Commit dc99065b5f97cc0410f88e3f90c7440531a55f9f
1 parent
ca735206
added flags computation optimization
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@34 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
13 changed files
with
3767 additions
and
88 deletions
Makefile
TODO
1 | -- optimize translated cache chaining (DLL PLT like system) | |
2 | -- improved 16 bit support | |
3 | -- optimize inverse flags propagation (easy by generating intermediate | |
4 | - micro operation array). | |
1 | +- optimize translated cache chaining (DLL PLT-like system) | |
2 | +- 64 bit syscalls | |
5 | 3 | - signals |
6 | 4 | - threads |
7 | 5 | - make it self runnable (use same trick as ld.so : include its own relocator and libc) |
6 | +- improved 16 bit support | |
8 | 7 | - fix FPU exceptions (in particular: gen_op_fpush not before mem load) |
9 | -- tests | ... | ... |
dis-asm.h
0 → 100644
1 | +/* Interface between the opcode library and its callers. | |
2 | + Written by Cygnus Support, 1993. | |
3 | + | |
4 | + The opcode library (libopcodes.a) provides instruction decoders for | |
5 | + a large variety of instruction sets, callable with an identical | |
6 | + interface, for making instruction-processing programs more independent | |
7 | + of the instruction set being processed. */ | |
8 | + | |
9 | +#ifndef DIS_ASM_H | |
10 | +#define DIS_ASM_H | |
11 | + | |
12 | +#include <stdio.h> | |
13 | +#include "bfd.h" | |
14 | + | |
15 | +typedef int (*fprintf_ftype) PARAMS((FILE*, const char*, ...)); | |
16 | + | |
17 | +enum dis_insn_type { | |
18 | + dis_noninsn, /* Not a valid instruction */ | |
19 | + dis_nonbranch, /* Not a branch instruction */ | |
20 | + dis_branch, /* Unconditional branch */ | |
21 | + dis_condbranch, /* Conditional branch */ | |
22 | + dis_jsr, /* Jump to subroutine */ | |
23 | + dis_condjsr, /* Conditional jump to subroutine */ | |
24 | + dis_dref, /* Data reference instruction */ | |
25 | + dis_dref2 /* Two data references in instruction */ | |
26 | +}; | |
27 | + | |
28 | +/* This struct is passed into the instruction decoding routine, | |
29 | + and is passed back out into each callback. The various fields are used | |
30 | + for conveying information from your main routine into your callbacks, | |
31 | + for passing information into the instruction decoders (such as the | |
32 | + addresses of the callback functions), or for passing information | |
33 | + back from the instruction decoders to their callers. | |
34 | + | |
35 | + It must be initialized before it is first passed; this can be done | |
36 | + by hand, or using one of the initialization macros below. */ | |
37 | + | |
38 | +typedef struct disassemble_info { | |
39 | + fprintf_ftype fprintf_func; | |
40 | + FILE *stream; | |
41 | + PTR application_data; | |
42 | + | |
43 | + /* Target description. We could replace this with a pointer to the bfd, | |
44 | + but that would require one. There currently isn't any such requirement | |
45 | + so to avoid introducing one we record these explicitly. */ | |
46 | + /* The bfd_flavour. This can be bfd_target_unknown_flavour. */ | |
47 | + enum bfd_flavour flavour; | |
48 | + /* The bfd_arch value. */ | |
49 | + enum bfd_architecture arch; | |
50 | + /* The bfd_mach value. */ | |
51 | + unsigned long mach; | |
52 | + /* Endianness (for bi-endian cpus). Mono-endian cpus can ignore this. */ | |
53 | + enum bfd_endian endian; | |
54 | + | |
55 | + /* An array of pointers to symbols either at the location being disassembled | |
56 | + or at the start of the function being disassembled. The array is sorted | |
57 | + so that the first symbol is intended to be the one used. The others are | |
58 | + present for any misc. purposes. This is not set reliably, but if it is | |
59 | + not NULL, it is correct. */ | |
60 | + asymbol **symbols; | |
61 | + /* Number of symbols in array. */ | |
62 | + int num_symbols; | |
63 | + | |
64 | + /* For use by the disassembler. | |
65 | + The top 16 bits are reserved for public use (and are documented here). | |
66 | + The bottom 16 bits are for the internal use of the disassembler. */ | |
67 | + unsigned long flags; | |
68 | +#define INSN_HAS_RELOC 0x80000000 | |
69 | + PTR private_data; | |
70 | + | |
71 | + /* Function used to get bytes to disassemble. MEMADDR is the | |
72 | + address of the stuff to be disassembled, MYADDR is the address to | |
73 | + put the bytes in, and LENGTH is the number of bytes to read. | |
74 | + INFO is a pointer to this struct. | |
75 | + Returns an errno value or 0 for success. */ | |
76 | + int (*read_memory_func) | |
77 | + PARAMS ((bfd_vma memaddr, bfd_byte *myaddr, int length, | |
78 | + struct disassemble_info *info)); | |
79 | + | |
80 | + /* Function which should be called if we get an error that we can't | |
81 | + recover from. STATUS is the errno value from read_memory_func and | |
82 | + MEMADDR is the address that we were trying to read. INFO is a | |
83 | + pointer to this struct. */ | |
84 | + void (*memory_error_func) | |
85 | + PARAMS ((int status, bfd_vma memaddr, struct disassemble_info *info)); | |
86 | + | |
87 | + /* Function called to print ADDR. */ | |
88 | + void (*print_address_func) | |
89 | + PARAMS ((bfd_vma addr, struct disassemble_info *info)); | |
90 | + | |
91 | + /* Function called to determine if there is a symbol at the given ADDR. | |
92 | + If there is, the function returns 1, otherwise it returns 0. | |
93 | + This is used by ports which support an overlay manager where | |
94 | + the overlay number is held in the top part of an address. In | |
95 | + some circumstances we want to include the overlay number in the | |
96 | + address, (normally because there is a symbol associated with | |
97 | + that address), but sometimes we want to mask out the overlay bits. */ | |
98 | + int (* symbol_at_address_func) | |
99 | + PARAMS ((bfd_vma addr, struct disassemble_info * info)); | |
100 | + | |
101 | + /* These are for buffer_read_memory. */ | |
102 | + bfd_byte *buffer; | |
103 | + bfd_vma buffer_vma; | |
104 | + int buffer_length; | |
105 | + | |
106 | + /* This variable may be set by the instruction decoder. It suggests | |
107 | + the number of bytes objdump should display on a single line. If | |
108 | + the instruction decoder sets this, it should always set it to | |
109 | + the same value in order to get reasonable looking output. */ | |
110 | + int bytes_per_line; | |
111 | + | |
112 | + /* the next two variables control the way objdump displays the raw data */ | |
113 | + /* For example, if bytes_per_line is 8 and bytes_per_chunk is 4, the */ | |
114 | + /* output will look like this: | |
115 | + 00: 00000000 00000000 | |
116 | + with the chunks displayed according to "display_endian". */ | |
117 | + int bytes_per_chunk; | |
118 | + enum bfd_endian display_endian; | |
119 | + | |
120 | + /* Results from instruction decoders. Not all decoders yet support | |
121 | + this information. This info is set each time an instruction is | |
122 | + decoded, and is only valid for the last such instruction. | |
123 | + | |
124 | + To determine whether this decoder supports this information, set | |
125 | + insn_info_valid to 0, decode an instruction, then check it. */ | |
126 | + | |
127 | + char insn_info_valid; /* Branch info has been set. */ | |
128 | + char branch_delay_insns; /* How many sequential insn's will run before | |
129 | + a branch takes effect. (0 = normal) */ | |
130 | + char data_size; /* Size of data reference in insn, in bytes */ | |
131 | + enum dis_insn_type insn_type; /* Type of instruction */ | |
132 | + bfd_vma target; /* Target address of branch or dref, if known; | |
133 | + zero if unknown. */ | |
134 | + bfd_vma target2; /* Second target address for dref2 */ | |
135 | + | |
136 | +} disassemble_info; | |
137 | + | |
138 | + | |
139 | +/* Standard disassemblers. Disassemble one instruction at the given | |
140 | + target address. Return number of bytes processed. */ | |
141 | +typedef int (*disassembler_ftype) | |
142 | + PARAMS((bfd_vma, disassemble_info *)); | |
143 | + | |
144 | +extern int print_insn_big_mips PARAMS ((bfd_vma, disassemble_info*)); | |
145 | +extern int print_insn_little_mips PARAMS ((bfd_vma, disassemble_info*)); | |
146 | +extern int print_insn_i386 PARAMS ((bfd_vma, disassemble_info*)); | |
147 | +extern int print_insn_m68k PARAMS ((bfd_vma, disassemble_info*)); | |
148 | +extern int print_insn_z8001 PARAMS ((bfd_vma, disassemble_info*)); | |
149 | +extern int print_insn_z8002 PARAMS ((bfd_vma, disassemble_info*)); | |
150 | +extern int print_insn_h8300 PARAMS ((bfd_vma, disassemble_info*)); | |
151 | +extern int print_insn_h8300h PARAMS ((bfd_vma, disassemble_info*)); | |
152 | +extern int print_insn_h8300s PARAMS ((bfd_vma, disassemble_info*)); | |
153 | +extern int print_insn_h8500 PARAMS ((bfd_vma, disassemble_info*)); | |
154 | +extern int print_insn_alpha PARAMS ((bfd_vma, disassemble_info*)); | |
155 | +extern disassembler_ftype arc_get_disassembler PARAMS ((int, int)); | |
156 | +extern int print_insn_big_arm PARAMS ((bfd_vma, disassemble_info*)); | |
157 | +extern int print_insn_little_arm PARAMS ((bfd_vma, disassemble_info*)); | |
158 | +extern int print_insn_sparc PARAMS ((bfd_vma, disassemble_info*)); | |
159 | +extern int print_insn_big_a29k PARAMS ((bfd_vma, disassemble_info*)); | |
160 | +extern int print_insn_little_a29k PARAMS ((bfd_vma, disassemble_info*)); | |
161 | +extern int print_insn_i960 PARAMS ((bfd_vma, disassemble_info*)); | |
162 | +extern int print_insn_sh PARAMS ((bfd_vma, disassemble_info*)); | |
163 | +extern int print_insn_shl PARAMS ((bfd_vma, disassemble_info*)); | |
164 | +extern int print_insn_hppa PARAMS ((bfd_vma, disassemble_info*)); | |
165 | +extern int print_insn_m32r PARAMS ((bfd_vma, disassemble_info*)); | |
166 | +extern int print_insn_m88k PARAMS ((bfd_vma, disassemble_info*)); | |
167 | +extern int print_insn_mn10200 PARAMS ((bfd_vma, disassemble_info*)); | |
168 | +extern int print_insn_mn10300 PARAMS ((bfd_vma, disassemble_info*)); | |
169 | +extern int print_insn_ns32k PARAMS ((bfd_vma, disassemble_info*)); | |
170 | +extern int print_insn_big_powerpc PARAMS ((bfd_vma, disassemble_info*)); | |
171 | +extern int print_insn_little_powerpc PARAMS ((bfd_vma, disassemble_info*)); | |
172 | +extern int print_insn_rs6000 PARAMS ((bfd_vma, disassemble_info*)); | |
173 | +extern int print_insn_w65 PARAMS ((bfd_vma, disassemble_info*)); | |
174 | +extern int print_insn_d10v PARAMS ((bfd_vma, disassemble_info*)); | |
175 | +extern int print_insn_v850 PARAMS ((bfd_vma, disassemble_info*)); | |
176 | +extern int print_insn_tic30 PARAMS ((bfd_vma, disassemble_info*)); | |
177 | + | |
178 | +/* Fetch the disassembler for a given BFD, if that support is available. */ | |
179 | +extern disassembler_ftype disassembler PARAMS ((bfd *)); | |
180 | + | |
181 | + | |
182 | +/* This block of definitions is for particular callers who read instructions | |
183 | + into a buffer before calling the instruction decoder. */ | |
184 | + | |
185 | +/* Here is a function which callers may wish to use for read_memory_func. | |
186 | + It gets bytes from a buffer. */ | |
187 | +extern int buffer_read_memory | |
188 | + PARAMS ((bfd_vma, bfd_byte *, int, struct disassemble_info *)); | |
189 | + | |
190 | +/* This function goes with buffer_read_memory. | |
191 | + It prints a message using info->fprintf_func and info->stream. */ | |
192 | +extern void perror_memory PARAMS ((int, bfd_vma, struct disassemble_info *)); | |
193 | + | |
194 | + | |
195 | +/* Just print the address in hex. This is included for completeness even | |
196 | + though both GDB and objdump provide their own (to print symbolic | |
197 | + addresses). */ | |
198 | +extern void generic_print_address | |
199 | + PARAMS ((bfd_vma, struct disassemble_info *)); | |
200 | + | |
201 | +/* Always true. */ | |
202 | +extern int generic_symbol_at_address | |
203 | + PARAMS ((bfd_vma, struct disassemble_info *)); | |
204 | + | |
205 | +/* Macro to initialize a disassemble_info struct. This should be called | |
206 | + by all applications creating such a struct. */ | |
207 | +#define INIT_DISASSEMBLE_INFO(INFO, STREAM, FPRINTF_FUNC) \ | |
208 | + (INFO).flavour = bfd_target_unknown_flavour, \ | |
209 | + (INFO).arch = bfd_arch_unknown, \ | |
210 | + (INFO).mach = 0, \ | |
211 | + (INFO).endian = BFD_ENDIAN_UNKNOWN, \ | |
212 | + INIT_DISASSEMBLE_INFO_NO_ARCH(INFO, STREAM, FPRINTF_FUNC) | |
213 | + | |
214 | +/* Call this macro to initialize only the internal variables for the | |
215 | + disassembler. Architecture dependent things such as byte order, or machine | |
216 | + variant are not touched by this macro. This makes things much easier for | |
217 | + GDB which must initialize these things seperatly. */ | |
218 | + | |
219 | +#define INIT_DISASSEMBLE_INFO_NO_ARCH(INFO, STREAM, FPRINTF_FUNC) \ | |
220 | + (INFO).fprintf_func = (FPRINTF_FUNC), \ | |
221 | + (INFO).stream = (STREAM), \ | |
222 | + (INFO).symbols = NULL, \ | |
223 | + (INFO).num_symbols = 0, \ | |
224 | + (INFO).buffer = NULL, \ | |
225 | + (INFO).buffer_vma = 0, \ | |
226 | + (INFO).buffer_length = 0, \ | |
227 | + (INFO).read_memory_func = buffer_read_memory, \ | |
228 | + (INFO).memory_error_func = perror_memory, \ | |
229 | + (INFO).print_address_func = generic_print_address, \ | |
230 | + (INFO).symbol_at_address_func = generic_symbol_at_address, \ | |
231 | + (INFO).flags = 0, \ | |
232 | + (INFO).bytes_per_line = 0, \ | |
233 | + (INFO).bytes_per_chunk = 0, \ | |
234 | + (INFO).display_endian = BFD_ENDIAN_UNKNOWN, \ | |
235 | + (INFO).insn_info_valid = 0 | |
236 | + | |
237 | +#endif /* ! defined (DIS_ASM_H) */ | ... | ... |
dis-buf.c
0 → 100644
1 | +/* Disassemble from a buffer, for GNU. | |
2 | + Copyright (C) 1993, 1994 Free Software Foundation, Inc. | |
3 | + | |
4 | +This program is free software; you can redistribute it and/or modify | |
5 | +it under the terms of the GNU General Public License as published by | |
6 | +the Free Software Foundation; either version 2 of the License, or | |
7 | +(at your option) any later version. | |
8 | + | |
9 | +This program is distributed in the hope that it will be useful, | |
10 | +but WITHOUT ANY WARRANTY; without even the implied warranty of | |
11 | +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
12 | +GNU General Public License for more details. | |
13 | + | |
14 | +You should have received a copy of the GNU General Public License | |
15 | +along with this program; if not, write to the Free Software | |
16 | +Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ | |
17 | + | |
18 | +#include "dis-asm.h" | |
19 | +#include <errno.h> | |
20 | + | |
21 | +/* Get LENGTH bytes from info's buffer, at target address memaddr. | |
22 | + Transfer them to myaddr. */ | |
23 | +int | |
24 | +buffer_read_memory (memaddr, myaddr, length, info) | |
25 | + bfd_vma memaddr; | |
26 | + bfd_byte *myaddr; | |
27 | + int length; | |
28 | + struct disassemble_info *info; | |
29 | +{ | |
30 | + if (memaddr < info->buffer_vma | |
31 | + || memaddr + length > info->buffer_vma + info->buffer_length) | |
32 | + /* Out of bounds. Use EIO because GDB uses it. */ | |
33 | + return EIO; | |
34 | + memcpy (myaddr, info->buffer + (memaddr - info->buffer_vma), length); | |
35 | + return 0; | |
36 | +} | |
37 | + | |
38 | +/* Print an error message. We can assume that this is in response to | |
39 | + an error return from buffer_read_memory. */ | |
40 | +void | |
41 | +perror_memory (status, memaddr, info) | |
42 | + int status; | |
43 | + bfd_vma memaddr; | |
44 | + struct disassemble_info *info; | |
45 | +{ | |
46 | + if (status != EIO) | |
47 | + /* Can't happen. */ | |
48 | + (*info->fprintf_func) (info->stream, "Unknown error %d\n", status); | |
49 | + else | |
50 | + /* Actually, address between memaddr and memaddr + len was | |
51 | + out of bounds. */ | |
52 | + (*info->fprintf_func) (info->stream, | |
53 | + "Address 0x%x is out of bounds.\n", memaddr); | |
54 | +} | |
55 | + | |
56 | +/* This could be in a separate file, to save miniscule amounts of space | |
57 | + in statically linked executables. */ | |
58 | + | |
59 | +/* Just print the address is hex. This is included for completeness even | |
60 | + though both GDB and objdump provide their own (to print symbolic | |
61 | + addresses). */ | |
62 | + | |
63 | +void | |
64 | +generic_print_address (addr, info) | |
65 | + bfd_vma addr; | |
66 | + struct disassemble_info *info; | |
67 | +{ | |
68 | + (*info->fprintf_func) (info->stream, "0x%x", addr); | |
69 | +} | |
70 | + | |
71 | +/* Just return the given address. */ | |
72 | + | |
73 | +int | |
74 | +generic_symbol_at_address (addr, info) | |
75 | + bfd_vma addr; | |
76 | + struct disassemble_info * info; | |
77 | +{ | |
78 | + return 1; | |
79 | +} | ... | ... |
dyngen.c
... | ... | @@ -28,7 +28,7 @@ |
28 | 28 | #include "thunk.h" |
29 | 29 | |
30 | 30 | /* all dynamically generated functions begin with this code */ |
31 | -#define OP_PREFIX "op" | |
31 | +#define OP_PREFIX "op_" | |
32 | 32 | |
33 | 33 | int elf_must_swap(Elf32_Ehdr *h) |
34 | 34 | { |
... | ... | @@ -201,7 +201,7 @@ int strstart(const char *str, const char *val, const char **ptr) |
201 | 201 | /* generate op code */ |
202 | 202 | void gen_code(const char *name, unsigned long offset, unsigned long size, |
203 | 203 | FILE *outfile, uint8_t *text, void *relocs, int nb_relocs, int reloc_sh_type, |
204 | - Elf32_Sym *symtab, char *strtab) | |
204 | + Elf32_Sym *symtab, char *strtab, int gen_switch) | |
205 | 205 | { |
206 | 206 | int copy_size = 0; |
207 | 207 | uint8_t *p_start, *p_end; |
... | ... | @@ -258,8 +258,6 @@ void gen_code(const char *name, unsigned long offset, unsigned long size, |
258 | 258 | if (n >= MAX_ARGS) |
259 | 259 | error("too many arguments in %s", name); |
260 | 260 | args_present[n - 1] = 1; |
261 | - } else { | |
262 | - fprintf(outfile, "extern char %s;\n", sym_name); | |
263 | 261 | } |
264 | 262 | } |
265 | 263 | } |
... | ... | @@ -274,8 +272,6 @@ void gen_code(const char *name, unsigned long offset, unsigned long size, |
274 | 272 | if (n >= MAX_ARGS) |
275 | 273 | error("too many arguments in %s", name); |
276 | 274 | args_present[n - 1] = 1; |
277 | - } else { | |
278 | - fprintf(outfile, "extern char %s;\n", sym_name); | |
279 | 275 | } |
280 | 276 | } |
281 | 277 | } |
... | ... | @@ -289,33 +285,59 @@ void gen_code(const char *name, unsigned long offset, unsigned long size, |
289 | 285 | error("inconsistent argument numbering in %s", name); |
290 | 286 | } |
291 | 287 | |
292 | - /* output C code */ | |
293 | - fprintf(outfile, "extern void %s();\n", name); | |
294 | - fprintf(outfile, "static inline void gen_%s(", name); | |
295 | - if (nb_args == 0) { | |
296 | - fprintf(outfile, "void"); | |
297 | - } else { | |
298 | - for(i = 0; i < nb_args; i++) { | |
299 | - if (i != 0) | |
300 | - fprintf(outfile, ", "); | |
301 | - fprintf(outfile, "long param%d", i + 1); | |
288 | + if (gen_switch) { | |
289 | + | |
290 | + /* output C code */ | |
291 | + fprintf(outfile, "case INDEX_%s: {\n", name); | |
292 | + if (nb_args > 0) { | |
293 | + fprintf(outfile, " long "); | |
294 | + for(i = 0; i < nb_args; i++) { | |
295 | + if (i != 0) | |
296 | + fprintf(outfile, ", "); | |
297 | + fprintf(outfile, "param%d", i + 1); | |
298 | + } | |
299 | + fprintf(outfile, ";\n"); | |
302 | 300 | } |
303 | - } | |
304 | - fprintf(outfile, ")\n"); | |
305 | - fprintf(outfile, "{\n"); | |
306 | - fprintf(outfile, " memcpy(gen_code_ptr, &%s, %d);\n", name, copy_size); | |
307 | - | |
308 | - /* patch relocations */ | |
309 | - switch(e_machine) { | |
310 | - case EM_386: | |
311 | - { | |
301 | + fprintf(outfile, " extern void %s();\n", name); | |
302 | + | |
303 | + if (reloc_sh_type == SHT_REL) { | |
312 | 304 | Elf32_Rel *rel; |
313 | - char name[256]; | |
314 | - int type; | |
315 | - long addend; | |
316 | 305 | for(i = 0, rel = relocs;i < nb_relocs; i++, rel++) { |
317 | 306 | if (rel->r_offset >= offset && rel->r_offset < offset + copy_size) { |
318 | 307 | sym_name = strtab + symtab[ELF32_R_SYM(rel->r_info)].st_name; |
308 | + if (!strstart(sym_name, "__op_param", &p)) { | |
309 | + fprintf(outfile, "extern char %s;\n", sym_name); | |
310 | + } | |
311 | + } | |
312 | + } | |
313 | + } else { | |
314 | + Elf32_Rela *rel; | |
315 | + for(i = 0, rel = relocs;i < nb_relocs; i++, rel++) { | |
316 | + if (rel->r_offset >= offset && rel->r_offset < offset + copy_size) { | |
317 | + sym_name = strtab + symtab[ELF32_R_SYM(rel->r_info)].st_name; | |
318 | + if (!strstart(sym_name, "__op_param", &p)) { | |
319 | + fprintf(outfile, "extern char %s;\n", sym_name); | |
320 | + } | |
321 | + } | |
322 | + } | |
323 | + } | |
324 | + | |
325 | + fprintf(outfile, " memcpy(gen_code_ptr, &%s, %d);\n", name, copy_size); | |
326 | + for(i = 0; i < nb_args; i++) { | |
327 | + fprintf(outfile, " param%d = *opparam_ptr++;\n", i + 1); | |
328 | + } | |
329 | + | |
330 | + /* patch relocations */ | |
331 | + switch(e_machine) { | |
332 | + case EM_386: | |
333 | + { | |
334 | + Elf32_Rel *rel; | |
335 | + char name[256]; | |
336 | + int type; | |
337 | + long addend; | |
338 | + for(i = 0, rel = relocs;i < nb_relocs; i++, rel++) { | |
339 | + if (rel->r_offset >= offset && rel->r_offset < offset + copy_size) { | |
340 | + sym_name = strtab + symtab[ELF32_R_SYM(rel->r_info)].st_name; | |
319 | 341 | if (strstart(sym_name, "__op_param", &p)) { |
320 | 342 | snprintf(name, sizeof(name), "param%s", p); |
321 | 343 | } else { |
... | ... | @@ -336,20 +358,38 @@ void gen_code(const char *name, unsigned long offset, unsigned long size, |
336 | 358 | error("unsupported i386 relocation (%d)", type); |
337 | 359 | } |
338 | 360 | } |
361 | + } | |
362 | + } | |
363 | + break; | |
364 | + default: | |
365 | + error("unsupported CPU for relocations (%d)", e_machine); | |
366 | + } | |
367 | + fprintf(outfile, " gen_code_ptr += %d;\n", copy_size); | |
368 | + fprintf(outfile, "}\n"); | |
369 | + fprintf(outfile, "break;\n\n"); | |
370 | + } else { | |
371 | + fprintf(outfile, "static inline void gen_%s(", name); | |
372 | + if (nb_args == 0) { | |
373 | + fprintf(outfile, "void"); | |
374 | + } else { | |
375 | + for(i = 0; i < nb_args; i++) { | |
376 | + if (i != 0) | |
377 | + fprintf(outfile, ", "); | |
378 | + fprintf(outfile, "long param%d", i + 1); | |
339 | 379 | } |
340 | 380 | } |
341 | - break; | |
342 | - default: | |
343 | - error("unsupported CPU for relocations (%d)", e_machine); | |
381 | + fprintf(outfile, ")\n"); | |
382 | + fprintf(outfile, "{\n"); | |
383 | + for(i = 0; i < nb_args; i++) { | |
384 | + fprintf(outfile, " *gen_opparam_ptr++ = param%d;\n", i + 1); | |
385 | + } | |
386 | + fprintf(outfile, " *gen_opc_ptr++ = INDEX_%s;\n", name); | |
387 | + fprintf(outfile, "}\n\n"); | |
344 | 388 | } |
345 | - | |
346 | - | |
347 | - fprintf(outfile, " gen_code_ptr += %d;\n", copy_size); | |
348 | - fprintf(outfile, "}\n\n"); | |
349 | 389 | } |
350 | 390 | |
351 | 391 | /* load an elf object file */ |
352 | -int load_elf(const char *filename, FILE *outfile) | |
392 | +int load_elf(const char *filename, FILE *outfile, int do_print_enum) | |
353 | 393 | { |
354 | 394 | int fd; |
355 | 395 | Elf32_Ehdr ehdr; |
... | ... | @@ -476,23 +516,77 @@ int load_elf(const char *filename, FILE *outfile) |
476 | 516 | error("unsupported CPU (e_machine=%d)", e_machine); |
477 | 517 | } |
478 | 518 | |
479 | - fprintf(outfile, "#include \"gen-%s.h\"\n\n", cpu_name); | |
519 | + if (do_print_enum) { | |
520 | + fprintf(outfile, "DEF(end)\n"); | |
521 | + for(i = 0, sym = symtab; i < nb_syms; i++, sym++) { | |
522 | + const char *name, *p; | |
523 | + name = strtab + sym->st_name; | |
524 | + if (strstart(name, OP_PREFIX, &p)) { | |
525 | + fprintf(outfile, "DEF(%s)\n", p); | |
526 | + } | |
527 | + } | |
528 | + } else { | |
529 | + /* generate big code generation switch */ | |
530 | +fprintf(outfile, | |
531 | +"int dyngen_code(uint8_t *gen_code_buf,\n" | |
532 | +" const uint16_t *opc_buf, const uint32_t *opparam_buf)\n" | |
533 | +"{\n" | |
534 | +" uint8_t *gen_code_ptr;\n" | |
535 | +" const uint16_t *opc_ptr;\n" | |
536 | +" const uint32_t *opparam_ptr;\n" | |
537 | +" gen_code_ptr = gen_code_buf;\n" | |
538 | +" opc_ptr = opc_buf;\n" | |
539 | +" opparam_ptr = opparam_buf;\n" | |
540 | +" for(;;) {\n" | |
541 | +" switch(*opc_ptr++) {\n" | |
542 | +); | |
480 | 543 | |
481 | - for(i = 0, sym = symtab; i < nb_syms; i++, sym++) { | |
482 | - const char *name; | |
483 | - name = strtab + sym->st_name; | |
484 | - if (strstart(name, "op_", NULL) || | |
485 | - strstart(name, "op1_", NULL) || | |
486 | - strstart(name, "op2_", NULL) || | |
487 | - strstart(name, "op3_", NULL)) { | |
544 | + for(i = 0, sym = symtab; i < nb_syms; i++, sym++) { | |
545 | + const char *name; | |
546 | + name = strtab + sym->st_name; | |
547 | + if (strstart(name, OP_PREFIX, NULL)) { | |
488 | 548 | #if 0 |
489 | - printf("%4d: %s pos=0x%08x len=%d\n", | |
490 | - i, name, sym->st_value, sym->st_size); | |
549 | + printf("%4d: %s pos=0x%08x len=%d\n", | |
550 | + i, name, sym->st_value, sym->st_size); | |
491 | 551 | #endif |
492 | - if (sym->st_shndx != (text_sec - shdr)) | |
493 | - error("invalid section for opcode (0x%x)", sym->st_shndx); | |
494 | - gen_code(name, sym->st_value, sym->st_size, outfile, | |
495 | - text, relocs, nb_relocs, reloc_sh_type, symtab, strtab); | |
552 | + if (sym->st_shndx != (text_sec - shdr)) | |
553 | + error("invalid section for opcode (0x%x)", sym->st_shndx); | |
554 | + gen_code(name, sym->st_value, sym->st_size, outfile, | |
555 | + text, relocs, nb_relocs, reloc_sh_type, symtab, strtab, 1); | |
556 | + } | |
557 | + } | |
558 | + | |
559 | +fprintf(outfile, | |
560 | +" default:\n" | |
561 | +" goto the_end;\n" | |
562 | +" }\n" | |
563 | +" }\n" | |
564 | +" the_end:\n" | |
565 | +); | |
566 | + | |
567 | +/* generate a return */ | |
568 | + switch(e_machine) { | |
569 | + case EM_386: | |
570 | + fprintf(outfile, "*gen_code_ptr++ = 0xc3; /* ret */\n"); | |
571 | + break; | |
572 | + default: | |
573 | + error("no return generation for cpu '%s'", cpu_name); | |
574 | + } | |
575 | + | |
576 | + fprintf(outfile, "return gen_code_ptr - gen_code_buf;\n"); | |
577 | + fprintf(outfile, "}\n\n"); | |
578 | + | |
579 | +/* generate gen_xxx functions */ | |
580 | +/* XXX: suppress the use of these functions to simplify code */ | |
581 | + for(i = 0, sym = symtab; i < nb_syms; i++, sym++) { | |
582 | + const char *name; | |
583 | + name = strtab + sym->st_name; | |
584 | + if (strstart(name, OP_PREFIX, NULL)) { | |
585 | + if (sym->st_shndx != (text_sec - shdr)) | |
586 | + error("invalid section for opcode (0x%x)", sym->st_shndx); | |
587 | + gen_code(name, sym->st_value, sym->st_size, outfile, | |
588 | + text, relocs, nb_relocs, reloc_sh_type, symtab, strtab, 0); | |
589 | + } | |
496 | 590 | } |
497 | 591 | } |
498 | 592 | |
... | ... | @@ -503,20 +597,23 @@ int load_elf(const char *filename, FILE *outfile) |
503 | 597 | void usage(void) |
504 | 598 | { |
505 | 599 | printf("dyngen (c) 2003 Fabrice Bellard\n" |
506 | - "usage: dyngen [-o outfile] objfile\n" | |
507 | - "Generate a dynamic code generator from an object file\n"); | |
600 | + "usage: dyngen [-o outfile] [-c] objfile\n" | |
601 | + "Generate a dynamic code generator from an object file\n" | |
602 | + "-c output enum of operations\n" | |
603 | + ); | |
508 | 604 | exit(1); |
509 | 605 | } |
510 | 606 | |
511 | 607 | int main(int argc, char **argv) |
512 | 608 | { |
513 | - int c; | |
609 | + int c, do_print_enum; | |
514 | 610 | const char *filename, *outfilename; |
515 | 611 | FILE *outfile; |
516 | 612 | |
517 | 613 | outfilename = "out.c"; |
614 | + do_print_enum = 0; | |
518 | 615 | for(;;) { |
519 | - c = getopt(argc, argv, "ho:"); | |
616 | + c = getopt(argc, argv, "ho:c"); | |
520 | 617 | if (c == -1) |
521 | 618 | break; |
522 | 619 | switch(c) { |
... | ... | @@ -526,6 +623,9 @@ int main(int argc, char **argv) |
526 | 623 | case 'o': |
527 | 624 | outfilename = optarg; |
528 | 625 | break; |
626 | + case 'c': | |
627 | + do_print_enum = 1; | |
628 | + break; | |
529 | 629 | } |
530 | 630 | } |
531 | 631 | if (optind >= argc) |
... | ... | @@ -534,7 +634,7 @@ int main(int argc, char **argv) |
534 | 634 | outfile = fopen(outfilename, "w"); |
535 | 635 | if (!outfile) |
536 | 636 | error("could not open '%s'", outfilename); |
537 | - load_elf(filename, outfile); | |
637 | + load_elf(filename, outfile, do_print_enum); | |
538 | 638 | fclose(outfile); |
539 | 639 | return 0; |
540 | 640 | } | ... | ... |
exec-i386.c
gen-i386.h
i386-dis.c
0 → 100644
1 | +/* Print i386 instructions for GDB, the GNU debugger. | |
2 | + Copyright (C) 1988, 89, 91, 93, 94, 95, 96, 97, 1998 | |
3 | + Free Software Foundation, Inc. | |
4 | + | |
5 | +This file is part of GDB. | |
6 | + | |
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 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, | |
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. | |
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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ | |
20 | + | |
21 | +/* | |
22 | + * 80386 instruction printer by Pace Willisson (pace@prep.ai.mit.edu) | |
23 | + * July 1988 | |
24 | + * modified by John Hassey (hassey@dg-rtp.dg.com) | |
25 | + */ | |
26 | + | |
27 | +/* | |
28 | + * The main tables describing the instructions is essentially a copy | |
29 | + * of the "Opcode Map" chapter (Appendix A) of the Intel 80386 | |
30 | + * Programmers Manual. Usually, there is a capital letter, followed | |
31 | + * by a small letter. The capital letter tell the addressing mode, | |
32 | + * and the small letter tells about the operand size. Refer to | |
33 | + * the Intel manual for details. | |
34 | + */ | |
35 | + | |
36 | +#include "dis-asm.h" | |
37 | + | |
38 | +#define MAXLEN 20 | |
39 | + | |
40 | +#include <setjmp.h> | |
41 | + | |
42 | +static int fetch_data PARAMS ((struct disassemble_info *, bfd_byte *)); | |
43 | + | |
44 | +struct dis_private | |
45 | +{ | |
46 | + /* Points to first byte not fetched. */ | |
47 | + bfd_byte *max_fetched; | |
48 | + bfd_byte the_buffer[MAXLEN]; | |
49 | + bfd_vma insn_start; | |
50 | + jmp_buf bailout; | |
51 | +}; | |
52 | + | |
53 | +/* Make sure that bytes from INFO->PRIVATE_DATA->BUFFER (inclusive) | |
54 | + to ADDR (exclusive) are valid. Returns 1 for success, longjmps | |
55 | + on error. */ | |
56 | +#define FETCH_DATA(info, addr) \ | |
57 | + ((addr) <= ((struct dis_private *)(info->private_data))->max_fetched \ | |
58 | + ? 1 : fetch_data ((info), (addr))) | |
59 | + | |
60 | +static int | |
61 | +fetch_data (info, addr) | |
62 | + struct disassemble_info *info; | |
63 | + bfd_byte *addr; | |
64 | +{ | |
65 | + int status; | |
66 | + struct dis_private *priv = (struct dis_private *)info->private_data; | |
67 | + bfd_vma start = priv->insn_start + (priv->max_fetched - priv->the_buffer); | |
68 | + | |
69 | + status = (*info->read_memory_func) (start, | |
70 | + priv->max_fetched, | |
71 | + addr - priv->max_fetched, | |
72 | + info); | |
73 | + if (status != 0) | |
74 | + { | |
75 | + (*info->memory_error_func) (status, start, info); | |
76 | + longjmp (priv->bailout, 1); | |
77 | + } | |
78 | + else | |
79 | + priv->max_fetched = addr; | |
80 | + return 1; | |
81 | +} | |
82 | + | |
83 | +#define Eb OP_E, b_mode | |
84 | +#define indirEb OP_indirE, b_mode | |
85 | +#define Gb OP_G, b_mode | |
86 | +#define Ev OP_E, v_mode | |
87 | +#define indirEv OP_indirE, v_mode | |
88 | +#define Ew OP_E, w_mode | |
89 | +#define Ma OP_E, v_mode | |
90 | +#define M OP_E, 0 | |
91 | +#define Mp OP_E, 0 /* ? */ | |
92 | +#define Gv OP_G, v_mode | |
93 | +#define Gw OP_G, w_mode | |
94 | +#define Rw OP_rm, w_mode | |
95 | +#define Rd OP_rm, d_mode | |
96 | +#define Ib OP_I, b_mode | |
97 | +#define sIb OP_sI, b_mode /* sign extened byte */ | |
98 | +#define Iv OP_I, v_mode | |
99 | +#define Iw OP_I, w_mode | |
100 | +#define Jb OP_J, b_mode | |
101 | +#define Jv OP_J, v_mode | |
102 | +#if 0 | |
103 | +#define ONE OP_ONE, 0 | |
104 | +#endif | |
105 | +#define Cd OP_C, d_mode | |
106 | +#define Dd OP_D, d_mode | |
107 | +#define Td OP_T, d_mode | |
108 | + | |
109 | +#define eAX OP_REG, eAX_reg | |
110 | +#define eBX OP_REG, eBX_reg | |
111 | +#define eCX OP_REG, eCX_reg | |
112 | +#define eDX OP_REG, eDX_reg | |
113 | +#define eSP OP_REG, eSP_reg | |
114 | +#define eBP OP_REG, eBP_reg | |
115 | +#define eSI OP_REG, eSI_reg | |
116 | +#define eDI OP_REG, eDI_reg | |
117 | +#define AL OP_REG, al_reg | |
118 | +#define CL OP_REG, cl_reg | |
119 | +#define DL OP_REG, dl_reg | |
120 | +#define BL OP_REG, bl_reg | |
121 | +#define AH OP_REG, ah_reg | |
122 | +#define CH OP_REG, ch_reg | |
123 | +#define DH OP_REG, dh_reg | |
124 | +#define BH OP_REG, bh_reg | |
125 | +#define AX OP_REG, ax_reg | |
126 | +#define DX OP_REG, dx_reg | |
127 | +#define indirDX OP_REG, indir_dx_reg | |
128 | + | |
129 | +#define Sw OP_SEG, w_mode | |
130 | +#define Ap OP_DIR, lptr | |
131 | +#define Av OP_DIR, v_mode | |
132 | +#define Ob OP_OFF, b_mode | |
133 | +#define Ov OP_OFF, v_mode | |
134 | +#define Xb OP_DSSI, b_mode | |
135 | +#define Xv OP_DSSI, v_mode | |
136 | +#define Yb OP_ESDI, b_mode | |
137 | +#define Yv OP_ESDI, v_mode | |
138 | + | |
139 | +#define es OP_REG, es_reg | |
140 | +#define ss OP_REG, ss_reg | |
141 | +#define cs OP_REG, cs_reg | |
142 | +#define ds OP_REG, ds_reg | |
143 | +#define fs OP_REG, fs_reg | |
144 | +#define gs OP_REG, gs_reg | |
145 | + | |
146 | +#define MX OP_MMX, 0 | |
147 | +#define EM OP_EM, v_mode | |
148 | +#define MS OP_MS, b_mode | |
149 | + | |
150 | +typedef int (*op_rtn) PARAMS ((int bytemode, int aflag, int dflag)); | |
151 | + | |
152 | +static int OP_E PARAMS ((int, int, int)); | |
153 | +static int OP_G PARAMS ((int, int, int)); | |
154 | +static int OP_I PARAMS ((int, int, int)); | |
155 | +static int OP_indirE PARAMS ((int, int, int)); | |
156 | +static int OP_sI PARAMS ((int, int, int)); | |
157 | +static int OP_REG PARAMS ((int, int, int)); | |
158 | +static int OP_J PARAMS ((int, int, int)); | |
159 | +static int OP_DIR PARAMS ((int, int, int)); | |
160 | +static int OP_OFF PARAMS ((int, int, int)); | |
161 | +static int OP_ESDI PARAMS ((int, int, int)); | |
162 | +static int OP_DSSI PARAMS ((int, int, int)); | |
163 | +static int OP_SEG PARAMS ((int, int, int)); | |
164 | +static int OP_C PARAMS ((int, int, int)); | |
165 | +static int OP_D PARAMS ((int, int, int)); | |
166 | +static int OP_T PARAMS ((int, int, int)); | |
167 | +static int OP_rm PARAMS ((int, int, int)); | |
168 | +static int OP_ST PARAMS ((int, int, int)); | |
169 | +static int OP_STi PARAMS ((int, int, int)); | |
170 | +#if 0 | |
171 | +static int OP_ONE PARAMS ((int, int, int)); | |
172 | +#endif | |
173 | +static int OP_MMX PARAMS ((int, int, int)); | |
174 | +static int OP_EM PARAMS ((int, int, int)); | |
175 | +static int OP_MS PARAMS ((int, int, int)); | |
176 | + | |
177 | +static void append_prefix PARAMS ((void)); | |
178 | +static void set_op PARAMS ((int op)); | |
179 | +static void putop PARAMS ((char *template, int aflag, int dflag)); | |
180 | +static void dofloat PARAMS ((int aflag, int dflag)); | |
181 | +static int get16 PARAMS ((void)); | |
182 | +static int get32 PARAMS ((void)); | |
183 | +static void ckprefix PARAMS ((void)); | |
184 | + | |
185 | +#define b_mode 1 | |
186 | +#define v_mode 2 | |
187 | +#define w_mode 3 | |
188 | +#define d_mode 4 | |
189 | + | |
190 | +#define es_reg 100 | |
191 | +#define cs_reg 101 | |
192 | +#define ss_reg 102 | |
193 | +#define ds_reg 103 | |
194 | +#define fs_reg 104 | |
195 | +#define gs_reg 105 | |
196 | +#define eAX_reg 107 | |
197 | +#define eCX_reg 108 | |
198 | +#define eDX_reg 109 | |
199 | +#define eBX_reg 110 | |
200 | +#define eSP_reg 111 | |
201 | +#define eBP_reg 112 | |
202 | +#define eSI_reg 113 | |
203 | +#define eDI_reg 114 | |
204 | + | |
205 | +#define lptr 115 | |
206 | + | |
207 | +#define al_reg 116 | |
208 | +#define cl_reg 117 | |
209 | +#define dl_reg 118 | |
210 | +#define bl_reg 119 | |
211 | +#define ah_reg 120 | |
212 | +#define ch_reg 121 | |
213 | +#define dh_reg 122 | |
214 | +#define bh_reg 123 | |
215 | + | |
216 | +#define ax_reg 124 | |
217 | +#define cx_reg 125 | |
218 | +#define dx_reg 126 | |
219 | +#define bx_reg 127 | |
220 | +#define sp_reg 128 | |
221 | +#define bp_reg 129 | |
222 | +#define si_reg 130 | |
223 | +#define di_reg 131 | |
224 | + | |
225 | +#define indir_dx_reg 150 | |
226 | + | |
227 | +#define GRP1b NULL, NULL, 0 | |
228 | +#define GRP1S NULL, NULL, 1 | |
229 | +#define GRP1Ss NULL, NULL, 2 | |
230 | +#define GRP2b NULL, NULL, 3 | |
231 | +#define GRP2S NULL, NULL, 4 | |
232 | +#define GRP2b_one NULL, NULL, 5 | |
233 | +#define GRP2S_one NULL, NULL, 6 | |
234 | +#define GRP2b_cl NULL, NULL, 7 | |
235 | +#define GRP2S_cl NULL, NULL, 8 | |
236 | +#define GRP3b NULL, NULL, 9 | |
237 | +#define GRP3S NULL, NULL, 10 | |
238 | +#define GRP4 NULL, NULL, 11 | |
239 | +#define GRP5 NULL, NULL, 12 | |
240 | +#define GRP6 NULL, NULL, 13 | |
241 | +#define GRP7 NULL, NULL, 14 | |
242 | +#define GRP8 NULL, NULL, 15 | |
243 | +#define GRP9 NULL, NULL, 16 | |
244 | +#define GRP10 NULL, NULL, 17 | |
245 | +#define GRP11 NULL, NULL, 18 | |
246 | +#define GRP12 NULL, NULL, 19 | |
247 | + | |
248 | +#define FLOATCODE 50 | |
249 | +#define FLOAT NULL, NULL, FLOATCODE | |
250 | + | |
251 | +struct dis386 { | |
252 | + char *name; | |
253 | + op_rtn op1; | |
254 | + int bytemode1; | |
255 | + op_rtn op2; | |
256 | + int bytemode2; | |
257 | + op_rtn op3; | |
258 | + int bytemode3; | |
259 | +}; | |
260 | + | |
261 | +static struct dis386 dis386[] = { | |
262 | + /* 00 */ | |
263 | + { "addb", Eb, Gb }, | |
264 | + { "addS", Ev, Gv }, | |
265 | + { "addb", Gb, Eb }, | |
266 | + { "addS", Gv, Ev }, | |
267 | + { "addb", AL, Ib }, | |
268 | + { "addS", eAX, Iv }, | |
269 | + { "pushS", es }, | |
270 | + { "popS", es }, | |
271 | + /* 08 */ | |
272 | + { "orb", Eb, Gb }, | |
273 | + { "orS", Ev, Gv }, | |
274 | + { "orb", Gb, Eb }, | |
275 | + { "orS", Gv, Ev }, | |
276 | + { "orb", AL, Ib }, | |
277 | + { "orS", eAX, Iv }, | |
278 | + { "pushS", cs }, | |
279 | + { "(bad)" }, /* 0x0f extended opcode escape */ | |
280 | + /* 10 */ | |
281 | + { "adcb", Eb, Gb }, | |
282 | + { "adcS", Ev, Gv }, | |
283 | + { "adcb", Gb, Eb }, | |
284 | + { "adcS", Gv, Ev }, | |
285 | + { "adcb", AL, Ib }, | |
286 | + { "adcS", eAX, Iv }, | |
287 | + { "pushS", ss }, | |
288 | + { "popS", ss }, | |
289 | + /* 18 */ | |
290 | + { "sbbb", Eb, Gb }, | |
291 | + { "sbbS", Ev, Gv }, | |
292 | + { "sbbb", Gb, Eb }, | |
293 | + { "sbbS", Gv, Ev }, | |
294 | + { "sbbb", AL, Ib }, | |
295 | + { "sbbS", eAX, Iv }, | |
296 | + { "pushS", ds }, | |
297 | + { "popS", ds }, | |
298 | + /* 20 */ | |
299 | + { "andb", Eb, Gb }, | |
300 | + { "andS", Ev, Gv }, | |
301 | + { "andb", Gb, Eb }, | |
302 | + { "andS", Gv, Ev }, | |
303 | + { "andb", AL, Ib }, | |
304 | + { "andS", eAX, Iv }, | |
305 | + { "(bad)" }, /* SEG ES prefix */ | |
306 | + { "daa" }, | |
307 | + /* 28 */ | |
308 | + { "subb", Eb, Gb }, | |
309 | + { "subS", Ev, Gv }, | |
310 | + { "subb", Gb, Eb }, | |
311 | + { "subS", Gv, Ev }, | |
312 | + { "subb", AL, Ib }, | |
313 | + { "subS", eAX, Iv }, | |
314 | + { "(bad)" }, /* SEG CS prefix */ | |
315 | + { "das" }, | |
316 | + /* 30 */ | |
317 | + { "xorb", Eb, Gb }, | |
318 | + { "xorS", Ev, Gv }, | |
319 | + { "xorb", Gb, Eb }, | |
320 | + { "xorS", Gv, Ev }, | |
321 | + { "xorb", AL, Ib }, | |
322 | + { "xorS", eAX, Iv }, | |
323 | + { "(bad)" }, /* SEG SS prefix */ | |
324 | + { "aaa" }, | |
325 | + /* 38 */ | |
326 | + { "cmpb", Eb, Gb }, | |
327 | + { "cmpS", Ev, Gv }, | |
328 | + { "cmpb", Gb, Eb }, | |
329 | + { "cmpS", Gv, Ev }, | |
330 | + { "cmpb", AL, Ib }, | |
331 | + { "cmpS", eAX, Iv }, | |
332 | + { "(bad)" }, /* SEG DS prefix */ | |
333 | + { "aas" }, | |
334 | + /* 40 */ | |
335 | + { "incS", eAX }, | |
336 | + { "incS", eCX }, | |
337 | + { "incS", eDX }, | |
338 | + { "incS", eBX }, | |
339 | + { "incS", eSP }, | |
340 | + { "incS", eBP }, | |
341 | + { "incS", eSI }, | |
342 | + { "incS", eDI }, | |
343 | + /* 48 */ | |
344 | + { "decS", eAX }, | |
345 | + { "decS", eCX }, | |
346 | + { "decS", eDX }, | |
347 | + { "decS", eBX }, | |
348 | + { "decS", eSP }, | |
349 | + { "decS", eBP }, | |
350 | + { "decS", eSI }, | |
351 | + { "decS", eDI }, | |
352 | + /* 50 */ | |
353 | + { "pushS", eAX }, | |
354 | + { "pushS", eCX }, | |
355 | + { "pushS", eDX }, | |
356 | + { "pushS", eBX }, | |
357 | + { "pushS", eSP }, | |
358 | + { "pushS", eBP }, | |
359 | + { "pushS", eSI }, | |
360 | + { "pushS", eDI }, | |
361 | + /* 58 */ | |
362 | + { "popS", eAX }, | |
363 | + { "popS", eCX }, | |
364 | + { "popS", eDX }, | |
365 | + { "popS", eBX }, | |
366 | + { "popS", eSP }, | |
367 | + { "popS", eBP }, | |
368 | + { "popS", eSI }, | |
369 | + { "popS", eDI }, | |
370 | + /* 60 */ | |
371 | + { "pusha" }, | |
372 | + { "popa" }, | |
373 | + { "boundS", Gv, Ma }, | |
374 | + { "arpl", Ew, Gw }, | |
375 | + { "(bad)" }, /* seg fs */ | |
376 | + { "(bad)" }, /* seg gs */ | |
377 | + { "(bad)" }, /* op size prefix */ | |
378 | + { "(bad)" }, /* adr size prefix */ | |
379 | + /* 68 */ | |
380 | + { "pushS", Iv }, /* 386 book wrong */ | |
381 | + { "imulS", Gv, Ev, Iv }, | |
382 | + { "pushS", sIb }, /* push of byte really pushes 2 or 4 bytes */ | |
383 | + { "imulS", Gv, Ev, Ib }, | |
384 | + { "insb", Yb, indirDX }, | |
385 | + { "insS", Yv, indirDX }, | |
386 | + { "outsb", indirDX, Xb }, | |
387 | + { "outsS", indirDX, Xv }, | |
388 | + /* 70 */ | |
389 | + { "jo", Jb }, | |
390 | + { "jno", Jb }, | |
391 | + { "jb", Jb }, | |
392 | + { "jae", Jb }, | |
393 | + { "je", Jb }, | |
394 | + { "jne", Jb }, | |
395 | + { "jbe", Jb }, | |
396 | + { "ja", Jb }, | |
397 | + /* 78 */ | |
398 | + { "js", Jb }, | |
399 | + { "jns", Jb }, | |
400 | + { "jp", Jb }, | |
401 | + { "jnp", Jb }, | |
402 | + { "jl", Jb }, | |
403 | + { "jnl", Jb }, | |
404 | + { "jle", Jb }, | |
405 | + { "jg", Jb }, | |
406 | + /* 80 */ | |
407 | + { GRP1b }, | |
408 | + { GRP1S }, | |
409 | + { "(bad)" }, | |
410 | + { GRP1Ss }, | |
411 | + { "testb", Eb, Gb }, | |
412 | + { "testS", Ev, Gv }, | |
413 | + { "xchgb", Eb, Gb }, | |
414 | + { "xchgS", Ev, Gv }, | |
415 | + /* 88 */ | |
416 | + { "movb", Eb, Gb }, | |
417 | + { "movS", Ev, Gv }, | |
418 | + { "movb", Gb, Eb }, | |
419 | + { "movS", Gv, Ev }, | |
420 | + { "movS", Ev, Sw }, | |
421 | + { "leaS", Gv, M }, | |
422 | + { "movS", Sw, Ev }, | |
423 | + { "popS", Ev }, | |
424 | + /* 90 */ | |
425 | + { "nop" }, | |
426 | + { "xchgS", eCX, eAX }, | |
427 | + { "xchgS", eDX, eAX }, | |
428 | + { "xchgS", eBX, eAX }, | |
429 | + { "xchgS", eSP, eAX }, | |
430 | + { "xchgS", eBP, eAX }, | |
431 | + { "xchgS", eSI, eAX }, | |
432 | + { "xchgS", eDI, eAX }, | |
433 | + /* 98 */ | |
434 | + { "cWtS" }, | |
435 | + { "cStd" }, | |
436 | + { "lcall", Ap }, | |
437 | + { "(bad)" }, /* fwait */ | |
438 | + { "pushf" }, | |
439 | + { "popf" }, | |
440 | + { "sahf" }, | |
441 | + { "lahf" }, | |
442 | + /* a0 */ | |
443 | + { "movb", AL, Ob }, | |
444 | + { "movS", eAX, Ov }, | |
445 | + { "movb", Ob, AL }, | |
446 | + { "movS", Ov, eAX }, | |
447 | + { "movsb", Yb, Xb }, | |
448 | + { "movsS", Yv, Xv }, | |
449 | + { "cmpsb", Yb, Xb }, | |
450 | + { "cmpsS", Yv, Xv }, | |
451 | + /* a8 */ | |
452 | + { "testb", AL, Ib }, | |
453 | + { "testS", eAX, Iv }, | |
454 | + { "stosb", Yb, AL }, | |
455 | + { "stosS", Yv, eAX }, | |
456 | + { "lodsb", AL, Xb }, | |
457 | + { "lodsS", eAX, Xv }, | |
458 | + { "scasb", AL, Yb }, | |
459 | + { "scasS", eAX, Yv }, | |
460 | + /* b0 */ | |
461 | + { "movb", AL, Ib }, | |
462 | + { "movb", CL, Ib }, | |
463 | + { "movb", DL, Ib }, | |
464 | + { "movb", BL, Ib }, | |
465 | + { "movb", AH, Ib }, | |
466 | + { "movb", CH, Ib }, | |
467 | + { "movb", DH, Ib }, | |
468 | + { "movb", BH, Ib }, | |
469 | + /* b8 */ | |
470 | + { "movS", eAX, Iv }, | |
471 | + { "movS", eCX, Iv }, | |
472 | + { "movS", eDX, Iv }, | |
473 | + { "movS", eBX, Iv }, | |
474 | + { "movS", eSP, Iv }, | |
475 | + { "movS", eBP, Iv }, | |
476 | + { "movS", eSI, Iv }, | |
477 | + { "movS", eDI, Iv }, | |
478 | + /* c0 */ | |
479 | + { GRP2b }, | |
480 | + { GRP2S }, | |
481 | + { "ret", Iw }, | |
482 | + { "ret" }, | |
483 | + { "lesS", Gv, Mp }, | |
484 | + { "ldsS", Gv, Mp }, | |
485 | + { "movb", Eb, Ib }, | |
486 | + { "movS", Ev, Iv }, | |
487 | + /* c8 */ | |
488 | + { "enter", Iw, Ib }, | |
489 | + { "leave" }, | |
490 | + { "lret", Iw }, | |
491 | + { "lret" }, | |
492 | + { "int3" }, | |
493 | + { "int", Ib }, | |
494 | + { "into" }, | |
495 | + { "iret" }, | |
496 | + /* d0 */ | |
497 | + { GRP2b_one }, | |
498 | + { GRP2S_one }, | |
499 | + { GRP2b_cl }, | |
500 | + { GRP2S_cl }, | |
501 | + { "aam", Ib }, | |
502 | + { "aad", Ib }, | |
503 | + { "(bad)" }, | |
504 | + { "xlat" }, | |
505 | + /* d8 */ | |
506 | + { FLOAT }, | |
507 | + { FLOAT }, | |
508 | + { FLOAT }, | |
509 | + { FLOAT }, | |
510 | + { FLOAT }, | |
511 | + { FLOAT }, | |
512 | + { FLOAT }, | |
513 | + { FLOAT }, | |
514 | + /* e0 */ | |
515 | + { "loopne", Jb }, | |
516 | + { "loope", Jb }, | |
517 | + { "loop", Jb }, | |
518 | + { "jCcxz", Jb }, | |
519 | + { "inb", AL, Ib }, | |
520 | + { "inS", eAX, Ib }, | |
521 | + { "outb", Ib, AL }, | |
522 | + { "outS", Ib, eAX }, | |
523 | + /* e8 */ | |
524 | + { "call", Av }, | |
525 | + { "jmp", Jv }, | |
526 | + { "ljmp", Ap }, | |
527 | + { "jmp", Jb }, | |
528 | + { "inb", AL, indirDX }, | |
529 | + { "inS", eAX, indirDX }, | |
530 | + { "outb", indirDX, AL }, | |
531 | + { "outS", indirDX, eAX }, | |
532 | + /* f0 */ | |
533 | + { "(bad)" }, /* lock prefix */ | |
534 | + { "(bad)" }, | |
535 | + { "(bad)" }, /* repne */ | |
536 | + { "(bad)" }, /* repz */ | |
537 | + { "hlt" }, | |
538 | + { "cmc" }, | |
539 | + { GRP3b }, | |
540 | + { GRP3S }, | |
541 | + /* f8 */ | |
542 | + { "clc" }, | |
543 | + { "stc" }, | |
544 | + { "cli" }, | |
545 | + { "sti" }, | |
546 | + { "cld" }, | |
547 | + { "std" }, | |
548 | + { GRP4 }, | |
549 | + { GRP5 }, | |
550 | +}; | |
551 | + | |
552 | +static struct dis386 dis386_twobyte[] = { | |
553 | + /* 00 */ | |
554 | + { GRP6 }, | |
555 | + { GRP7 }, | |
556 | + { "larS", Gv, Ew }, | |
557 | + { "lslS", Gv, Ew }, | |
558 | + { "(bad)" }, | |
559 | + { "(bad)" }, | |
560 | + { "clts" }, | |
561 | + { "(bad)" }, | |
562 | + /* 08 */ | |
563 | + { "invd" }, | |
564 | + { "wbinvd" }, | |
565 | + { "(bad)" }, { "ud2a" }, | |
566 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
567 | + /* 10 */ | |
568 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
569 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
570 | + /* 18 */ | |
571 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
572 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
573 | + /* 20 */ | |
574 | + /* these are all backward in appendix A of the intel book */ | |
575 | + { "movl", Rd, Cd }, | |
576 | + { "movl", Rd, Dd }, | |
577 | + { "movl", Cd, Rd }, | |
578 | + { "movl", Dd, Rd }, | |
579 | + { "movl", Rd, Td }, | |
580 | + { "(bad)" }, | |
581 | + { "movl", Td, Rd }, | |
582 | + { "(bad)" }, | |
583 | + /* 28 */ | |
584 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
585 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
586 | + /* 30 */ | |
587 | + { "wrmsr" }, { "rdtsc" }, { "rdmsr" }, { "rdpmc" }, | |
588 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
589 | + /* 38 */ | |
590 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
591 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
592 | + /* 40 */ | |
593 | + { "cmovo", Gv,Ev }, { "cmovno", Gv,Ev }, { "cmovb", Gv,Ev }, { "cmovae", Gv,Ev }, | |
594 | + { "cmove", Gv,Ev }, { "cmovne", Gv,Ev }, { "cmovbe", Gv,Ev }, { "cmova", Gv,Ev }, | |
595 | + /* 48 */ | |
596 | + { "cmovs", Gv,Ev }, { "cmovns", Gv,Ev }, { "cmovp", Gv,Ev }, { "cmovnp", Gv,Ev }, | |
597 | + { "cmovl", Gv,Ev }, { "cmovge", Gv,Ev }, { "cmovle", Gv,Ev }, { "cmovg", Gv,Ev }, | |
598 | + /* 50 */ | |
599 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
600 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
601 | + /* 58 */ | |
602 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
603 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
604 | + /* 60 */ | |
605 | + { "punpcklbw", MX, EM }, | |
606 | + { "punpcklwd", MX, EM }, | |
607 | + { "punpckldq", MX, EM }, | |
608 | + { "packsswb", MX, EM }, | |
609 | + { "pcmpgtb", MX, EM }, | |
610 | + { "pcmpgtw", MX, EM }, | |
611 | + { "pcmpgtd", MX, EM }, | |
612 | + { "packuswb", MX, EM }, | |
613 | + /* 68 */ | |
614 | + { "punpckhbw", MX, EM }, | |
615 | + { "punpckhwd", MX, EM }, | |
616 | + { "punpckhdq", MX, EM }, | |
617 | + { "packssdw", MX, EM }, | |
618 | + { "(bad)" }, { "(bad)" }, | |
619 | + { "movd", MX, Ev }, | |
620 | + { "movq", MX, EM }, | |
621 | + /* 70 */ | |
622 | + { "(bad)" }, | |
623 | + { GRP10 }, | |
624 | + { GRP11 }, | |
625 | + { GRP12 }, | |
626 | + { "pcmpeqb", MX, EM }, | |
627 | + { "pcmpeqw", MX, EM }, | |
628 | + { "pcmpeqd", MX, EM }, | |
629 | + { "emms" }, | |
630 | + /* 78 */ | |
631 | + { "(bad)" }, { "(bad)" }, { "(bad)" }, { "(bad)" }, | |
632 | + { "(bad)" }, { "(bad)" }, | |
633 | + { "movd", Ev, MX }, | |
634 | + { "movq", EM, MX }, | |
635 | + /* 80 */ | |
636 | + { "jo", Jv }, | |
637 | + { "jno", Jv }, | |
638 | + { "jb", Jv }, | |
639 | + { "jae", Jv }, | |
640 | + { "je", Jv }, | |
641 | + { "jne", Jv }, | |
642 | + { "jbe", Jv }, | |
643 | + { "ja", Jv }, | |
644 | + /* 88 */ | |
645 | + { "js", Jv }, | |
646 | + { "jns", Jv }, | |
647 | + { "jp", Jv }, | |
648 | + { "jnp", Jv }, | |
649 | + { "jl", Jv }, | |
650 | + { "jge", Jv }, | |
651 | + { "jle", Jv }, | |
652 | + { "jg", Jv }, | |
653 | + /* 90 */ | |
654 | + { "seto", Eb }, | |
655 | + { "setno", Eb }, | |
656 | + { "setb", Eb }, | |
657 | + { "setae", Eb }, | |
658 | + { "sete", Eb }, | |
659 | + { "setne", Eb }, | |
660 | + { "setbe", Eb }, | |
661 | + { "seta", Eb }, | |
662 | + /* 98 */ | |
663 | + { "sets", Eb }, | |
664 | + { "setns", Eb }, | |
665 | + { "setp", Eb }, | |
666 | + { "setnp", Eb }, | |
667 | + { "setl", Eb }, | |
668 | + { "setge", Eb }, | |
669 | + { "setle", Eb }, | |
670 | + { "setg", Eb }, | |
671 | + /* a0 */ | |
672 | + { "pushS", fs }, | |
673 | + { "popS", fs }, | |
674 | + { "cpuid" }, | |
675 | + { "btS", Ev, Gv }, | |
676 | + { "shldS", Ev, Gv, Ib }, | |
677 | + { "shldS", Ev, Gv, CL }, | |
678 | + { "(bad)" }, | |
679 | + { "(bad)" }, | |
680 | + /* a8 */ | |
681 | + { "pushS", gs }, | |
682 | + { "popS", gs }, | |
683 | + { "rsm" }, | |
684 | + { "btsS", Ev, Gv }, | |
685 | + { "shrdS", Ev, Gv, Ib }, | |
686 | + { "shrdS", Ev, Gv, CL }, | |
687 | + { "(bad)" }, | |
688 | + { "imulS", Gv, Ev }, | |
689 | + /* b0 */ | |
690 | + { "cmpxchgb", Eb, Gb }, | |
691 | + { "cmpxchgS", Ev, Gv }, | |
692 | + { "lssS", Gv, Mp }, /* 386 lists only Mp */ | |
693 | + { "btrS", Ev, Gv }, | |
694 | + { "lfsS", Gv, Mp }, /* 386 lists only Mp */ | |
695 | + { "lgsS", Gv, Mp }, /* 386 lists only Mp */ | |
696 | + { "movzbS", Gv, Eb }, | |
697 | + { "movzwS", Gv, Ew }, | |
698 | + /* b8 */ | |
699 | + { "ud2b" }, | |
700 | + { "(bad)" }, | |
701 | + { GRP8 }, | |
702 | + { "btcS", Ev, Gv }, | |
703 | + { "bsfS", Gv, Ev }, | |
704 | + { "bsrS", Gv, Ev }, | |
705 | + { "movsbS", Gv, Eb }, | |
706 | + { "movswS", Gv, Ew }, | |
707 | + /* c0 */ | |
708 | + { "xaddb", Eb, Gb }, | |
709 | + { "xaddS", Ev, Gv }, | |
710 | + { "(bad)" }, | |
711 | + { "(bad)" }, | |
712 | + { "(bad)" }, | |
713 | + { "(bad)" }, | |
714 | + { "(bad)" }, | |
715 | + { GRP9 }, | |
716 | + /* c8 */ | |
717 | + { "bswap", eAX }, | |
718 | + { "bswap", eCX }, | |
719 | + { "bswap", eDX }, | |
720 | + { "bswap", eBX }, | |
721 | + { "bswap", eSP }, | |
722 | + { "bswap", eBP }, | |
723 | + { "bswap", eSI }, | |
724 | + { "bswap", eDI }, | |
725 | + /* d0 */ | |
726 | + { "(bad)" }, | |
727 | + { "psrlw", MX, EM }, | |
728 | + { "psrld", MX, EM }, | |
729 | + { "psrlq", MX, EM }, | |
730 | + { "(bad)" }, | |
731 | + { "pmullw", MX, EM }, | |
732 | + { "(bad)" }, { "(bad)" }, | |
733 | + /* d8 */ | |
734 | + { "psubusb", MX, EM }, | |
735 | + { "psubusw", MX, EM }, | |
736 | + { "(bad)" }, | |
737 | + { "pand", MX, EM }, | |
738 | + { "paddusb", MX, EM }, | |
739 | + { "paddusw", MX, EM }, | |
740 | + { "(bad)" }, | |
741 | + { "pandn", MX, EM }, | |
742 | + /* e0 */ | |
743 | + { "(bad)" }, | |
744 | + { "psraw", MX, EM }, | |
745 | + { "psrad", MX, EM }, | |
746 | + { "(bad)" }, | |
747 | + { "(bad)" }, | |
748 | + { "pmulhw", MX, EM }, | |
749 | + { "(bad)" }, { "(bad)" }, | |
750 | + /* e8 */ | |
751 | + { "psubsb", MX, EM }, | |
752 | + { "psubsw", MX, EM }, | |
753 | + { "(bad)" }, | |
754 | + { "por", MX, EM }, | |
755 | + { "paddsb", MX, EM }, | |
756 | + { "paddsw", MX, EM }, | |
757 | + { "(bad)" }, | |
758 | + { "pxor", MX, EM }, | |
759 | + /* f0 */ | |
760 | + { "(bad)" }, | |
761 | + { "psllw", MX, EM }, | |
762 | + { "pslld", MX, EM }, | |
763 | + { "psllq", MX, EM }, | |
764 | + { "(bad)" }, | |
765 | + { "pmaddwd", MX, EM }, | |
766 | + { "(bad)" }, { "(bad)" }, | |
767 | + /* f8 */ | |
768 | + { "psubb", MX, EM }, | |
769 | + { "psubw", MX, EM }, | |
770 | + { "psubd", MX, EM }, | |
771 | + { "(bad)" }, | |
772 | + { "paddb", MX, EM }, | |
773 | + { "paddw", MX, EM }, | |
774 | + { "paddd", MX, EM }, | |
775 | + { "(bad)" } | |
776 | +}; | |
777 | + | |
778 | +static const unsigned char onebyte_has_modrm[256] = { | |
779 | + 1,1,1,1,0,0,0,0,1,1,1,1,0,0,0,0, | |
780 | + 1,1,1,1,0,0,0,0,1,1,1,1,0,0,0,0, | |
781 | + 1,1,1,1,0,0,0,0,1,1,1,1,0,0,0,0, | |
782 | + 1,1,1,1,0,0,0,0,1,1,1,1,0,0,0,0, | |
783 | + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, | |
784 | + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, | |
785 | + 0,0,1,1,0,0,0,0,0,1,0,1,0,0,0,0, | |
786 | + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, | |
787 | + 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, | |
788 | + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, | |
789 | + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, | |
790 | + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, | |
791 | + 1,1,0,0,1,1,1,1,0,0,0,0,0,0,0,0, | |
792 | + 1,1,1,1,0,0,0,0,1,1,1,1,1,1,1,1, | |
793 | + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, | |
794 | + 0,0,0,0,0,0,1,1,0,0,0,0,0,0,1,1 | |
795 | +}; | |
796 | + | |
797 | +static const unsigned char twobyte_has_modrm[256] = { | |
798 | + /* 00 */ 1,1,1,1,0,0,0,0,0,0,0,0,0,0,0,0, /* 0f */ | |
799 | + /* 10 */ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 1f */ | |
800 | + /* 20 */ 1,1,1,1,1,1,1,1,0,0,0,0,0,0,0,0, /* 2f */ | |
801 | + /* 30 */ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 3f */ | |
802 | + /* 40 */ 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, /* 4f */ | |
803 | + /* 50 */ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 5f */ | |
804 | + /* 60 */ 1,1,1,1,1,1,1,1,1,1,1,1,0,0,1,1, /* 6f */ | |
805 | + /* 70 */ 0,1,1,1,1,1,1,0,0,0,0,0,0,0,1,1, /* 7f */ | |
806 | + /* 80 */ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 8f */ | |
807 | + /* 90 */ 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, /* 9f */ | |
808 | + /* a0 */ 0,0,0,1,1,1,1,1,0,0,0,1,1,1,1,1, /* af */ | |
809 | + /* b0 */ 1,1,1,1,1,1,1,1,0,0,1,1,1,1,1,1, /* bf */ | |
810 | + /* c0 */ 1,1,1,1,1,1,1,1,0,0,0,0,0,0,0,0, /* cf */ | |
811 | + /* d0 */ 0,1,1,1,0,1,0,0,1,1,0,1,1,1,0,1, /* df */ | |
812 | + /* e0 */ 0,1,1,0,0,1,0,0,1,1,0,1,1,1,0,1, /* ef */ | |
813 | + /* f0 */ 0,1,1,1,0,1,0,0,1,1,1,0,1,1,1,0 /* ff */ | |
814 | +}; | |
815 | + | |
816 | +static char obuf[100]; | |
817 | +static char *obufp; | |
818 | +static char scratchbuf[100]; | |
819 | +static unsigned char *start_codep; | |
820 | +static unsigned char *codep; | |
821 | +static disassemble_info *the_info; | |
822 | +static int mod; | |
823 | +static int rm; | |
824 | +static int reg; | |
825 | +static void oappend PARAMS ((char *s)); | |
826 | + | |
827 | +static char *names32[]={ | |
828 | + "%eax","%ecx","%edx","%ebx", "%esp","%ebp","%esi","%edi", | |
829 | +}; | |
830 | +static char *names16[] = { | |
831 | + "%ax","%cx","%dx","%bx","%sp","%bp","%si","%di", | |
832 | +}; | |
833 | +static char *names8[] = { | |
834 | + "%al","%cl","%dl","%bl","%ah","%ch","%dh","%bh", | |
835 | +}; | |
836 | +static char *names_seg[] = { | |
837 | + "%es","%cs","%ss","%ds","%fs","%gs","%?","%?", | |
838 | +}; | |
839 | +static char *index16[] = { | |
840 | + "bx+si","bx+di","bp+si","bp+di","si","di","bp","bx" | |
841 | +}; | |
842 | + | |
843 | +static struct dis386 grps[][8] = { | |
844 | + /* GRP1b */ | |
845 | + { | |
846 | + { "addb", Eb, Ib }, | |
847 | + { "orb", Eb, Ib }, | |
848 | + { "adcb", Eb, Ib }, | |
849 | + { "sbbb", Eb, Ib }, | |
850 | + { "andb", Eb, Ib }, | |
851 | + { "subb", Eb, Ib }, | |
852 | + { "xorb", Eb, Ib }, | |
853 | + { "cmpb", Eb, Ib } | |
854 | + }, | |
855 | + /* GRP1S */ | |
856 | + { | |
857 | + { "addS", Ev, Iv }, | |
858 | + { "orS", Ev, Iv }, | |
859 | + { "adcS", Ev, Iv }, | |
860 | + { "sbbS", Ev, Iv }, | |
861 | + { "andS", Ev, Iv }, | |
862 | + { "subS", Ev, Iv }, | |
863 | + { "xorS", Ev, Iv }, | |
864 | + { "cmpS", Ev, Iv } | |
865 | + }, | |
866 | + /* GRP1Ss */ | |
867 | + { | |
868 | + { "addS", Ev, sIb }, | |
869 | + { "orS", Ev, sIb }, | |
870 | + { "adcS", Ev, sIb }, | |
871 | + { "sbbS", Ev, sIb }, | |
872 | + { "andS", Ev, sIb }, | |
873 | + { "subS", Ev, sIb }, | |
874 | + { "xorS", Ev, sIb }, | |
875 | + { "cmpS", Ev, sIb } | |
876 | + }, | |
877 | + /* GRP2b */ | |
878 | + { | |
879 | + { "rolb", Eb, Ib }, | |
880 | + { "rorb", Eb, Ib }, | |
881 | + { "rclb", Eb, Ib }, | |
882 | + { "rcrb", Eb, Ib }, | |
883 | + { "shlb", Eb, Ib }, | |
884 | + { "shrb", Eb, Ib }, | |
885 | + { "(bad)" }, | |
886 | + { "sarb", Eb, Ib }, | |
887 | + }, | |
888 | + /* GRP2S */ | |
889 | + { | |
890 | + { "rolS", Ev, Ib }, | |
891 | + { "rorS", Ev, Ib }, | |
892 | + { "rclS", Ev, Ib }, | |
893 | + { "rcrS", Ev, Ib }, | |
894 | + { "shlS", Ev, Ib }, | |
895 | + { "shrS", Ev, Ib }, | |
896 | + { "(bad)" }, | |
897 | + { "sarS", Ev, Ib }, | |
898 | + }, | |
899 | + /* GRP2b_one */ | |
900 | + { | |
901 | + { "rolb", Eb }, | |
902 | + { "rorb", Eb }, | |
903 | + { "rclb", Eb }, | |
904 | + { "rcrb", Eb }, | |
905 | + { "shlb", Eb }, | |
906 | + { "shrb", Eb }, | |
907 | + { "(bad)" }, | |
908 | + { "sarb", Eb }, | |
909 | + }, | |
910 | + /* GRP2S_one */ | |
911 | + { | |
912 | + { "rolS", Ev }, | |
913 | + { "rorS", Ev }, | |
914 | + { "rclS", Ev }, | |
915 | + { "rcrS", Ev }, | |
916 | + { "shlS", Ev }, | |
917 | + { "shrS", Ev }, | |
918 | + { "(bad)" }, | |
919 | + { "sarS", Ev }, | |
920 | + }, | |
921 | + /* GRP2b_cl */ | |
922 | + { | |
923 | + { "rolb", Eb, CL }, | |
924 | + { "rorb", Eb, CL }, | |
925 | + { "rclb", Eb, CL }, | |
926 | + { "rcrb", Eb, CL }, | |
927 | + { "shlb", Eb, CL }, | |
928 | + { "shrb", Eb, CL }, | |
929 | + { "(bad)" }, | |
930 | + { "sarb", Eb, CL }, | |
931 | + }, | |
932 | + /* GRP2S_cl */ | |
933 | + { | |
934 | + { "rolS", Ev, CL }, | |
935 | + { "rorS", Ev, CL }, | |
936 | + { "rclS", Ev, CL }, | |
937 | + { "rcrS", Ev, CL }, | |
938 | + { "shlS", Ev, CL }, | |
939 | + { "shrS", Ev, CL }, | |
940 | + { "(bad)" }, | |
941 | + { "sarS", Ev, CL } | |
942 | + }, | |
943 | + /* GRP3b */ | |
944 | + { | |
945 | + { "testb", Eb, Ib }, | |
946 | + { "(bad)", Eb }, | |
947 | + { "notb", Eb }, | |
948 | + { "negb", Eb }, | |
949 | + { "mulb", AL, Eb }, | |
950 | + { "imulb", AL, Eb }, | |
951 | + { "divb", AL, Eb }, | |
952 | + { "idivb", AL, Eb } | |
953 | + }, | |
954 | + /* GRP3S */ | |
955 | + { | |
956 | + { "testS", Ev, Iv }, | |
957 | + { "(bad)" }, | |
958 | + { "notS", Ev }, | |
959 | + { "negS", Ev }, | |
960 | + { "mulS", eAX, Ev }, | |
961 | + { "imulS", eAX, Ev }, | |
962 | + { "divS", eAX, Ev }, | |
963 | + { "idivS", eAX, Ev }, | |
964 | + }, | |
965 | + /* GRP4 */ | |
966 | + { | |
967 | + { "incb", Eb }, | |
968 | + { "decb", Eb }, | |
969 | + { "(bad)" }, | |
970 | + { "(bad)" }, | |
971 | + { "(bad)" }, | |
972 | + { "(bad)" }, | |
973 | + { "(bad)" }, | |
974 | + { "(bad)" }, | |
975 | + }, | |
976 | + /* GRP5 */ | |
977 | + { | |
978 | + { "incS", Ev }, | |
979 | + { "decS", Ev }, | |
980 | + { "call", indirEv }, | |
981 | + { "lcall", indirEv }, | |
982 | + { "jmp", indirEv }, | |
983 | + { "ljmp", indirEv }, | |
984 | + { "pushS", Ev }, | |
985 | + { "(bad)" }, | |
986 | + }, | |
987 | + /* GRP6 */ | |
988 | + { | |
989 | + { "sldt", Ew }, | |
990 | + { "str", Ew }, | |
991 | + { "lldt", Ew }, | |
992 | + { "ltr", Ew }, | |
993 | + { "verr", Ew }, | |
994 | + { "verw", Ew }, | |
995 | + { "(bad)" }, | |
996 | + { "(bad)" } | |
997 | + }, | |
998 | + /* GRP7 */ | |
999 | + { | |
1000 | + { "sgdt", Ew }, | |
1001 | + { "sidt", Ew }, | |
1002 | + { "lgdt", Ew }, | |
1003 | + { "lidt", Ew }, | |
1004 | + { "smsw", Ew }, | |
1005 | + { "(bad)" }, | |
1006 | + { "lmsw", Ew }, | |
1007 | + { "invlpg", Ew }, | |
1008 | + }, | |
1009 | + /* GRP8 */ | |
1010 | + { | |
1011 | + { "(bad)" }, | |
1012 | + { "(bad)" }, | |
1013 | + { "(bad)" }, | |
1014 | + { "(bad)" }, | |
1015 | + { "btS", Ev, Ib }, | |
1016 | + { "btsS", Ev, Ib }, | |
1017 | + { "btrS", Ev, Ib }, | |
1018 | + { "btcS", Ev, Ib }, | |
1019 | + }, | |
1020 | + /* GRP9 */ | |
1021 | + { | |
1022 | + { "(bad)" }, | |
1023 | + { "cmpxchg8b", Ev }, | |
1024 | + { "(bad)" }, | |
1025 | + { "(bad)" }, | |
1026 | + { "(bad)" }, | |
1027 | + { "(bad)" }, | |
1028 | + { "(bad)" }, | |
1029 | + { "(bad)" }, | |
1030 | + }, | |
1031 | + /* GRP10 */ | |
1032 | + { | |
1033 | + { "(bad)" }, | |
1034 | + { "(bad)" }, | |
1035 | + { "psrlw", MS, Ib }, | |
1036 | + { "(bad)" }, | |
1037 | + { "psraw", MS, Ib }, | |
1038 | + { "(bad)" }, | |
1039 | + { "psllw", MS, Ib }, | |
1040 | + { "(bad)" }, | |
1041 | + }, | |
1042 | + /* GRP11 */ | |
1043 | + { | |
1044 | + { "(bad)" }, | |
1045 | + { "(bad)" }, | |
1046 | + { "psrld", MS, Ib }, | |
1047 | + { "(bad)" }, | |
1048 | + { "psrad", MS, Ib }, | |
1049 | + { "(bad)" }, | |
1050 | + { "pslld", MS, Ib }, | |
1051 | + { "(bad)" }, | |
1052 | + }, | |
1053 | + /* GRP12 */ | |
1054 | + { | |
1055 | + { "(bad)" }, | |
1056 | + { "(bad)" }, | |
1057 | + { "psrlq", MS, Ib }, | |
1058 | + { "(bad)" }, | |
1059 | + { "(bad)" }, | |
1060 | + { "(bad)" }, | |
1061 | + { "psllq", MS, Ib }, | |
1062 | + { "(bad)" }, | |
1063 | + } | |
1064 | +}; | |
1065 | + | |
1066 | +#define PREFIX_REPZ 1 | |
1067 | +#define PREFIX_REPNZ 2 | |
1068 | +#define PREFIX_LOCK 4 | |
1069 | +#define PREFIX_CS 8 | |
1070 | +#define PREFIX_SS 0x10 | |
1071 | +#define PREFIX_DS 0x20 | |
1072 | +#define PREFIX_ES 0x40 | |
1073 | +#define PREFIX_FS 0x80 | |
1074 | +#define PREFIX_GS 0x100 | |
1075 | +#define PREFIX_DATA 0x200 | |
1076 | +#define PREFIX_ADR 0x400 | |
1077 | +#define PREFIX_FWAIT 0x800 | |
1078 | + | |
1079 | +static int prefixes; | |
1080 | + | |
1081 | +static void | |
1082 | +ckprefix () | |
1083 | +{ | |
1084 | + prefixes = 0; | |
1085 | + while (1) | |
1086 | + { | |
1087 | + FETCH_DATA (the_info, codep + 1); | |
1088 | + switch (*codep) | |
1089 | + { | |
1090 | + case 0xf3: | |
1091 | + prefixes |= PREFIX_REPZ; | |
1092 | + break; | |
1093 | + case 0xf2: | |
1094 | + prefixes |= PREFIX_REPNZ; | |
1095 | + break; | |
1096 | + case 0xf0: | |
1097 | + prefixes |= PREFIX_LOCK; | |
1098 | + break; | |
1099 | + case 0x2e: | |
1100 | + prefixes |= PREFIX_CS; | |
1101 | + break; | |
1102 | + case 0x36: | |
1103 | + prefixes |= PREFIX_SS; | |
1104 | + break; | |
1105 | + case 0x3e: | |
1106 | + prefixes |= PREFIX_DS; | |
1107 | + break; | |
1108 | + case 0x26: | |
1109 | + prefixes |= PREFIX_ES; | |
1110 | + break; | |
1111 | + case 0x64: | |
1112 | + prefixes |= PREFIX_FS; | |
1113 | + break; | |
1114 | + case 0x65: | |
1115 | + prefixes |= PREFIX_GS; | |
1116 | + break; | |
1117 | + case 0x66: | |
1118 | + prefixes |= PREFIX_DATA; | |
1119 | + break; | |
1120 | + case 0x67: | |
1121 | + prefixes |= PREFIX_ADR; | |
1122 | + break; | |
1123 | + case 0x9b: | |
1124 | + prefixes |= PREFIX_FWAIT; | |
1125 | + break; | |
1126 | + default: | |
1127 | + return; | |
1128 | + } | |
1129 | + codep++; | |
1130 | + } | |
1131 | +} | |
1132 | + | |
1133 | +static char op1out[100], op2out[100], op3out[100]; | |
1134 | +static int op_address[3], op_ad, op_index[3]; | |
1135 | +static int start_pc; | |
1136 | + | |
1137 | + | |
1138 | +/* | |
1139 | + * On the 386's of 1988, the maximum length of an instruction is 15 bytes. | |
1140 | + * (see topic "Redundant prefixes" in the "Differences from 8086" | |
1141 | + * section of the "Virtual 8086 Mode" chapter.) | |
1142 | + * 'pc' should be the address of this instruction, it will | |
1143 | + * be used to print the target address if this is a relative jump or call | |
1144 | + * The function returns the length of this instruction in bytes. | |
1145 | + */ | |
1146 | + | |
1147 | +int print_insn_x86 PARAMS ((bfd_vma pc, disassemble_info *info, int aflag, | |
1148 | + int dflag)); | |
1149 | +int | |
1150 | +print_insn_i386 (pc, info) | |
1151 | + bfd_vma pc; | |
1152 | + disassemble_info *info; | |
1153 | +{ | |
1154 | + if (info->mach == bfd_mach_i386_i386) | |
1155 | + return print_insn_x86 (pc, info, 1, 1); | |
1156 | + else if (info->mach == bfd_mach_i386_i8086) | |
1157 | + return print_insn_x86 (pc, info, 0, 0); | |
1158 | + else | |
1159 | + abort (); | |
1160 | +} | |
1161 | + | |
1162 | +int | |
1163 | +print_insn_x86 (pc, info, aflag, dflag) | |
1164 | + bfd_vma pc; | |
1165 | + disassemble_info *info; | |
1166 | + int aflag; | |
1167 | + int dflag; | |
1168 | +{ | |
1169 | + struct dis386 *dp; | |
1170 | + int i; | |
1171 | + int enter_instruction; | |
1172 | + char *first, *second, *third; | |
1173 | + int needcomma; | |
1174 | + unsigned char need_modrm; | |
1175 | + | |
1176 | + struct dis_private priv; | |
1177 | + bfd_byte *inbuf = priv.the_buffer; | |
1178 | + | |
1179 | + /* The output looks better if we put 5 bytes on a line, since that | |
1180 | + puts long word instructions on a single line. */ | |
1181 | + info->bytes_per_line = 5; | |
1182 | + | |
1183 | + info->private_data = (PTR) &priv; | |
1184 | + priv.max_fetched = priv.the_buffer; | |
1185 | + priv.insn_start = pc; | |
1186 | + if (setjmp (priv.bailout) != 0) | |
1187 | + /* Error return. */ | |
1188 | + return -1; | |
1189 | + | |
1190 | + obuf[0] = 0; | |
1191 | + op1out[0] = 0; | |
1192 | + op2out[0] = 0; | |
1193 | + op3out[0] = 0; | |
1194 | + | |
1195 | + op_index[0] = op_index[1] = op_index[2] = -1; | |
1196 | + | |
1197 | + the_info = info; | |
1198 | + start_pc = pc; | |
1199 | + start_codep = inbuf; | |
1200 | + codep = inbuf; | |
1201 | + | |
1202 | + ckprefix (); | |
1203 | + | |
1204 | + FETCH_DATA (info, codep + 1); | |
1205 | + if (*codep == 0xc8) | |
1206 | + enter_instruction = 1; | |
1207 | + else | |
1208 | + enter_instruction = 0; | |
1209 | + | |
1210 | + obufp = obuf; | |
1211 | + | |
1212 | + if (prefixes & PREFIX_REPZ) | |
1213 | + oappend ("repz "); | |
1214 | + if (prefixes & PREFIX_REPNZ) | |
1215 | + oappend ("repnz "); | |
1216 | + if (prefixes & PREFIX_LOCK) | |
1217 | + oappend ("lock "); | |
1218 | + | |
1219 | + if ((prefixes & PREFIX_FWAIT) | |
1220 | + && ((*codep < 0xd8) || (*codep > 0xdf))) | |
1221 | + { | |
1222 | + /* fwait not followed by floating point instruction */ | |
1223 | + (*info->fprintf_func) (info->stream, "fwait"); | |
1224 | + return (1); | |
1225 | + } | |
1226 | + | |
1227 | + if (prefixes & PREFIX_DATA) | |
1228 | + dflag ^= 1; | |
1229 | + | |
1230 | + if (prefixes & PREFIX_ADR) | |
1231 | + { | |
1232 | + aflag ^= 1; | |
1233 | + if (aflag) | |
1234 | + oappend ("addr32 "); | |
1235 | + else | |
1236 | + oappend ("addr16 "); | |
1237 | + } | |
1238 | + | |
1239 | + if (*codep == 0x0f) | |
1240 | + { | |
1241 | + FETCH_DATA (info, codep + 2); | |
1242 | + dp = &dis386_twobyte[*++codep]; | |
1243 | + need_modrm = twobyte_has_modrm[*codep]; | |
1244 | + } | |
1245 | + else | |
1246 | + { | |
1247 | + dp = &dis386[*codep]; | |
1248 | + need_modrm = onebyte_has_modrm[*codep]; | |
1249 | + } | |
1250 | + codep++; | |
1251 | + | |
1252 | + if (need_modrm) | |
1253 | + { | |
1254 | + FETCH_DATA (info, codep + 1); | |
1255 | + mod = (*codep >> 6) & 3; | |
1256 | + reg = (*codep >> 3) & 7; | |
1257 | + rm = *codep & 7; | |
1258 | + } | |
1259 | + | |
1260 | + if (dp->name == NULL && dp->bytemode1 == FLOATCODE) | |
1261 | + { | |
1262 | + dofloat (aflag, dflag); | |
1263 | + } | |
1264 | + else | |
1265 | + { | |
1266 | + if (dp->name == NULL) | |
1267 | + dp = &grps[dp->bytemode1][reg]; | |
1268 | + | |
1269 | + putop (dp->name, aflag, dflag); | |
1270 | + | |
1271 | + obufp = op1out; | |
1272 | + op_ad = 2; | |
1273 | + if (dp->op1) | |
1274 | + (*dp->op1)(dp->bytemode1, aflag, dflag); | |
1275 | + | |
1276 | + obufp = op2out; | |
1277 | + op_ad = 1; | |
1278 | + if (dp->op2) | |
1279 | + (*dp->op2)(dp->bytemode2, aflag, dflag); | |
1280 | + | |
1281 | + obufp = op3out; | |
1282 | + op_ad = 0; | |
1283 | + if (dp->op3) | |
1284 | + (*dp->op3)(dp->bytemode3, aflag, dflag); | |
1285 | + } | |
1286 | + | |
1287 | + obufp = obuf + strlen (obuf); | |
1288 | + for (i = strlen (obuf); i < 6; i++) | |
1289 | + oappend (" "); | |
1290 | + oappend (" "); | |
1291 | + (*info->fprintf_func) (info->stream, "%s", obuf); | |
1292 | + | |
1293 | + /* enter instruction is printed with operands in the | |
1294 | + * same order as the intel book; everything else | |
1295 | + * is printed in reverse order | |
1296 | + */ | |
1297 | + if (enter_instruction) | |
1298 | + { | |
1299 | + first = op1out; | |
1300 | + second = op2out; | |
1301 | + third = op3out; | |
1302 | + op_ad = op_index[0]; | |
1303 | + op_index[0] = op_index[2]; | |
1304 | + op_index[2] = op_ad; | |
1305 | + } | |
1306 | + else | |
1307 | + { | |
1308 | + first = op3out; | |
1309 | + second = op2out; | |
1310 | + third = op1out; | |
1311 | + } | |
1312 | + needcomma = 0; | |
1313 | + if (*first) | |
1314 | + { | |
1315 | + if (op_index[0] != -1) | |
1316 | + (*info->print_address_func) (op_address[op_index[0]], info); | |
1317 | + else | |
1318 | + (*info->fprintf_func) (info->stream, "%s", first); | |
1319 | + needcomma = 1; | |
1320 | + } | |
1321 | + if (*second) | |
1322 | + { | |
1323 | + if (needcomma) | |
1324 | + (*info->fprintf_func) (info->stream, ","); | |
1325 | + if (op_index[1] != -1) | |
1326 | + (*info->print_address_func) (op_address[op_index[1]], info); | |
1327 | + else | |
1328 | + (*info->fprintf_func) (info->stream, "%s", second); | |
1329 | + needcomma = 1; | |
1330 | + } | |
1331 | + if (*third) | |
1332 | + { | |
1333 | + if (needcomma) | |
1334 | + (*info->fprintf_func) (info->stream, ","); | |
1335 | + if (op_index[2] != -1) | |
1336 | + (*info->print_address_func) (op_address[op_index[2]], info); | |
1337 | + else | |
1338 | + (*info->fprintf_func) (info->stream, "%s", third); | |
1339 | + } | |
1340 | + return (codep - inbuf); | |
1341 | +} | |
1342 | + | |
1343 | +static char *float_mem[] = { | |
1344 | + /* d8 */ | |
1345 | + "fadds", | |
1346 | + "fmuls", | |
1347 | + "fcoms", | |
1348 | + "fcomps", | |
1349 | + "fsubs", | |
1350 | + "fsubrs", | |
1351 | + "fdivs", | |
1352 | + "fdivrs", | |
1353 | + /* d9 */ | |
1354 | + "flds", | |
1355 | + "(bad)", | |
1356 | + "fsts", | |
1357 | + "fstps", | |
1358 | + "fldenv", | |
1359 | + "fldcw", | |
1360 | + "fNstenv", | |
1361 | + "fNstcw", | |
1362 | + /* da */ | |
1363 | + "fiaddl", | |
1364 | + "fimull", | |
1365 | + "ficoml", | |
1366 | + "ficompl", | |
1367 | + "fisubl", | |
1368 | + "fisubrl", | |
1369 | + "fidivl", | |
1370 | + "fidivrl", | |
1371 | + /* db */ | |
1372 | + "fildl", | |
1373 | + "(bad)", | |
1374 | + "fistl", | |
1375 | + "fistpl", | |
1376 | + "(bad)", | |
1377 | + "fldt", | |
1378 | + "(bad)", | |
1379 | + "fstpt", | |
1380 | + /* dc */ | |
1381 | + "faddl", | |
1382 | + "fmull", | |
1383 | + "fcoml", | |
1384 | + "fcompl", | |
1385 | + "fsubl", | |
1386 | + "fsubrl", | |
1387 | + "fdivl", | |
1388 | + "fdivrl", | |
1389 | + /* dd */ | |
1390 | + "fldl", | |
1391 | + "(bad)", | |
1392 | + "fstl", | |
1393 | + "fstpl", | |
1394 | + "frstor", | |
1395 | + "(bad)", | |
1396 | + "fNsave", | |
1397 | + "fNstsw", | |
1398 | + /* de */ | |
1399 | + "fiadd", | |
1400 | + "fimul", | |
1401 | + "ficom", | |
1402 | + "ficomp", | |
1403 | + "fisub", | |
1404 | + "fisubr", | |
1405 | + "fidiv", | |
1406 | + "fidivr", | |
1407 | + /* df */ | |
1408 | + "fild", | |
1409 | + "(bad)", | |
1410 | + "fist", | |
1411 | + "fistp", | |
1412 | + "fbld", | |
1413 | + "fildll", | |
1414 | + "fbstp", | |
1415 | + "fistpll", | |
1416 | +}; | |
1417 | + | |
1418 | +#define ST OP_ST, 0 | |
1419 | +#define STi OP_STi, 0 | |
1420 | + | |
1421 | +#define FGRPd9_2 NULL, NULL, 0 | |
1422 | +#define FGRPd9_4 NULL, NULL, 1 | |
1423 | +#define FGRPd9_5 NULL, NULL, 2 | |
1424 | +#define FGRPd9_6 NULL, NULL, 3 | |
1425 | +#define FGRPd9_7 NULL, NULL, 4 | |
1426 | +#define FGRPda_5 NULL, NULL, 5 | |
1427 | +#define FGRPdb_4 NULL, NULL, 6 | |
1428 | +#define FGRPde_3 NULL, NULL, 7 | |
1429 | +#define FGRPdf_4 NULL, NULL, 8 | |
1430 | + | |
1431 | +static struct dis386 float_reg[][8] = { | |
1432 | + /* d8 */ | |
1433 | + { | |
1434 | + { "fadd", ST, STi }, | |
1435 | + { "fmul", ST, STi }, | |
1436 | + { "fcom", STi }, | |
1437 | + { "fcomp", STi }, | |
1438 | + { "fsub", ST, STi }, | |
1439 | + { "fsubr", ST, STi }, | |
1440 | + { "fdiv", ST, STi }, | |
1441 | + { "fdivr", ST, STi }, | |
1442 | + }, | |
1443 | + /* d9 */ | |
1444 | + { | |
1445 | + { "fld", STi }, | |
1446 | + { "fxch", STi }, | |
1447 | + { FGRPd9_2 }, | |
1448 | + { "(bad)" }, | |
1449 | + { FGRPd9_4 }, | |
1450 | + { FGRPd9_5 }, | |
1451 | + { FGRPd9_6 }, | |
1452 | + { FGRPd9_7 }, | |
1453 | + }, | |
1454 | + /* da */ | |
1455 | + { | |
1456 | + { "fcmovb", ST, STi }, | |
1457 | + { "fcmove", ST, STi }, | |
1458 | + { "fcmovbe",ST, STi }, | |
1459 | + { "fcmovu", ST, STi }, | |
1460 | + { "(bad)" }, | |
1461 | + { FGRPda_5 }, | |
1462 | + { "(bad)" }, | |
1463 | + { "(bad)" }, | |
1464 | + }, | |
1465 | + /* db */ | |
1466 | + { | |
1467 | + { "fcmovnb",ST, STi }, | |
1468 | + { "fcmovne",ST, STi }, | |
1469 | + { "fcmovnbe",ST, STi }, | |
1470 | + { "fcmovnu",ST, STi }, | |
1471 | + { FGRPdb_4 }, | |
1472 | + { "fucomi", ST, STi }, | |
1473 | + { "fcomi", ST, STi }, | |
1474 | + { "(bad)" }, | |
1475 | + }, | |
1476 | + /* dc */ | |
1477 | + { | |
1478 | + { "fadd", STi, ST }, | |
1479 | + { "fmul", STi, ST }, | |
1480 | + { "(bad)" }, | |
1481 | + { "(bad)" }, | |
1482 | + { "fsub", STi, ST }, | |
1483 | + { "fsubr", STi, ST }, | |
1484 | + { "fdiv", STi, ST }, | |
1485 | + { "fdivr", STi, ST }, | |
1486 | + }, | |
1487 | + /* dd */ | |
1488 | + { | |
1489 | + { "ffree", STi }, | |
1490 | + { "(bad)" }, | |
1491 | + { "fst", STi }, | |
1492 | + { "fstp", STi }, | |
1493 | + { "fucom", STi }, | |
1494 | + { "fucomp", STi }, | |
1495 | + { "(bad)" }, | |
1496 | + { "(bad)" }, | |
1497 | + }, | |
1498 | + /* de */ | |
1499 | + { | |
1500 | + { "faddp", STi, ST }, | |
1501 | + { "fmulp", STi, ST }, | |
1502 | + { "(bad)" }, | |
1503 | + { FGRPde_3 }, | |
1504 | + { "fsubp", STi, ST }, | |
1505 | + { "fsubrp", STi, ST }, | |
1506 | + { "fdivp", STi, ST }, | |
1507 | + { "fdivrp", STi, ST }, | |
1508 | + }, | |
1509 | + /* df */ | |
1510 | + { | |
1511 | + { "(bad)" }, | |
1512 | + { "(bad)" }, | |
1513 | + { "(bad)" }, | |
1514 | + { "(bad)" }, | |
1515 | + { FGRPdf_4 }, | |
1516 | + { "fucomip",ST, STi }, | |
1517 | + { "fcomip", ST, STi }, | |
1518 | + { "(bad)" }, | |
1519 | + }, | |
1520 | +}; | |
1521 | + | |
1522 | + | |
1523 | +static char *fgrps[][8] = { | |
1524 | + /* d9_2 0 */ | |
1525 | + { | |
1526 | + "fnop","(bad)","(bad)","(bad)","(bad)","(bad)","(bad)","(bad)", | |
1527 | + }, | |
1528 | + | |
1529 | + /* d9_4 1 */ | |
1530 | + { | |
1531 | + "fchs","fabs","(bad)","(bad)","ftst","fxam","(bad)","(bad)", | |
1532 | + }, | |
1533 | + | |
1534 | + /* d9_5 2 */ | |
1535 | + { | |
1536 | + "fld1","fldl2t","fldl2e","fldpi","fldlg2","fldln2","fldz","(bad)", | |
1537 | + }, | |
1538 | + | |
1539 | + /* d9_6 3 */ | |
1540 | + { | |
1541 | + "f2xm1","fyl2x","fptan","fpatan","fxtract","fprem1","fdecstp","fincstp", | |
1542 | + }, | |
1543 | + | |
1544 | + /* d9_7 4 */ | |
1545 | + { | |
1546 | + "fprem","fyl2xp1","fsqrt","fsincos","frndint","fscale","fsin","fcos", | |
1547 | + }, | |
1548 | + | |
1549 | + /* da_5 5 */ | |
1550 | + { | |
1551 | + "(bad)","fucompp","(bad)","(bad)","(bad)","(bad)","(bad)","(bad)", | |
1552 | + }, | |
1553 | + | |
1554 | + /* db_4 6 */ | |
1555 | + { | |
1556 | + "feni(287 only)","fdisi(287 only)","fNclex","fNinit", | |
1557 | + "fNsetpm(287 only)","(bad)","(bad)","(bad)", | |
1558 | + }, | |
1559 | + | |
1560 | + /* de_3 7 */ | |
1561 | + { | |
1562 | + "(bad)","fcompp","(bad)","(bad)","(bad)","(bad)","(bad)","(bad)", | |
1563 | + }, | |
1564 | + | |
1565 | + /* df_4 8 */ | |
1566 | + { | |
1567 | + "fNstsw","(bad)","(bad)","(bad)","(bad)","(bad)","(bad)","(bad)", | |
1568 | + }, | |
1569 | +}; | |
1570 | + | |
1571 | +static void | |
1572 | +dofloat (aflag, dflag) | |
1573 | + int aflag; | |
1574 | + int dflag; | |
1575 | +{ | |
1576 | + struct dis386 *dp; | |
1577 | + unsigned char floatop; | |
1578 | + | |
1579 | + floatop = codep[-1]; | |
1580 | + | |
1581 | + if (mod != 3) | |
1582 | + { | |
1583 | + putop (float_mem[(floatop - 0xd8) * 8 + reg], aflag, dflag); | |
1584 | + obufp = op1out; | |
1585 | + OP_E (v_mode, aflag, dflag); | |
1586 | + return; | |
1587 | + } | |
1588 | + codep++; | |
1589 | + | |
1590 | + dp = &float_reg[floatop - 0xd8][reg]; | |
1591 | + if (dp->name == NULL) | |
1592 | + { | |
1593 | + putop (fgrps[dp->bytemode1][rm], aflag, dflag); | |
1594 | + /* instruction fnstsw is only one with strange arg */ | |
1595 | + if (floatop == 0xdf | |
1596 | + && FETCH_DATA (the_info, codep + 1) | |
1597 | + && *codep == 0xe0) | |
1598 | + strcpy (op1out, "%eax"); | |
1599 | + } | |
1600 | + else | |
1601 | + { | |
1602 | + putop (dp->name, aflag, dflag); | |
1603 | + obufp = op1out; | |
1604 | + if (dp->op1) | |
1605 | + (*dp->op1)(dp->bytemode1, aflag, dflag); | |
1606 | + obufp = op2out; | |
1607 | + if (dp->op2) | |
1608 | + (*dp->op2)(dp->bytemode2, aflag, dflag); | |
1609 | + } | |
1610 | +} | |
1611 | + | |
1612 | +/* ARGSUSED */ | |
1613 | +static int | |
1614 | +OP_ST (ignore, aflag, dflag) | |
1615 | + int ignore; | |
1616 | + int aflag; | |
1617 | + int dflag; | |
1618 | +{ | |
1619 | + oappend ("%st"); | |
1620 | + return (0); | |
1621 | +} | |
1622 | + | |
1623 | +/* ARGSUSED */ | |
1624 | +static int | |
1625 | +OP_STi (ignore, aflag, dflag) | |
1626 | + int ignore; | |
1627 | + int aflag; | |
1628 | + int dflag; | |
1629 | +{ | |
1630 | + sprintf (scratchbuf, "%%st(%d)", rm); | |
1631 | + oappend (scratchbuf); | |
1632 | + return (0); | |
1633 | +} | |
1634 | + | |
1635 | + | |
1636 | +/* capital letters in template are macros */ | |
1637 | +static void | |
1638 | +putop (template, aflag, dflag) | |
1639 | + char *template; | |
1640 | + int aflag; | |
1641 | + int dflag; | |
1642 | +{ | |
1643 | + char *p; | |
1644 | + | |
1645 | + for (p = template; *p; p++) | |
1646 | + { | |
1647 | + switch (*p) | |
1648 | + { | |
1649 | + default: | |
1650 | + *obufp++ = *p; | |
1651 | + break; | |
1652 | + case 'C': /* For jcxz/jecxz */ | |
1653 | + if (aflag) | |
1654 | + *obufp++ = 'e'; | |
1655 | + break; | |
1656 | + case 'N': | |
1657 | + if ((prefixes & PREFIX_FWAIT) == 0) | |
1658 | + *obufp++ = 'n'; | |
1659 | + break; | |
1660 | + case 'S': | |
1661 | + /* operand size flag */ | |
1662 | + if (dflag) | |
1663 | + *obufp++ = 'l'; | |
1664 | + else | |
1665 | + *obufp++ = 'w'; | |
1666 | + break; | |
1667 | + case 'W': | |
1668 | + /* operand size flag for cwtl, cbtw */ | |
1669 | + if (dflag) | |
1670 | + *obufp++ = 'w'; | |
1671 | + else | |
1672 | + *obufp++ = 'b'; | |
1673 | + break; | |
1674 | + } | |
1675 | + } | |
1676 | + *obufp = 0; | |
1677 | +} | |
1678 | + | |
1679 | +static void | |
1680 | +oappend (s) | |
1681 | + char *s; | |
1682 | +{ | |
1683 | + strcpy (obufp, s); | |
1684 | + obufp += strlen (s); | |
1685 | + *obufp = 0; | |
1686 | +} | |
1687 | + | |
1688 | +static void | |
1689 | +append_prefix () | |
1690 | +{ | |
1691 | + if (prefixes & PREFIX_CS) | |
1692 | + oappend ("%cs:"); | |
1693 | + if (prefixes & PREFIX_DS) | |
1694 | + oappend ("%ds:"); | |
1695 | + if (prefixes & PREFIX_SS) | |
1696 | + oappend ("%ss:"); | |
1697 | + if (prefixes & PREFIX_ES) | |
1698 | + oappend ("%es:"); | |
1699 | + if (prefixes & PREFIX_FS) | |
1700 | + oappend ("%fs:"); | |
1701 | + if (prefixes & PREFIX_GS) | |
1702 | + oappend ("%gs:"); | |
1703 | +} | |
1704 | + | |
1705 | +static int | |
1706 | +OP_indirE (bytemode, aflag, dflag) | |
1707 | + int bytemode; | |
1708 | + int aflag; | |
1709 | + int dflag; | |
1710 | +{ | |
1711 | + oappend ("*"); | |
1712 | + return OP_E (bytemode, aflag, dflag); | |
1713 | +} | |
1714 | + | |
1715 | +static int | |
1716 | +OP_E (bytemode, aflag, dflag) | |
1717 | + int bytemode; | |
1718 | + int aflag; | |
1719 | + int dflag; | |
1720 | +{ | |
1721 | + int disp; | |
1722 | + | |
1723 | + /* skip mod/rm byte */ | |
1724 | + codep++; | |
1725 | + | |
1726 | + if (mod == 3) | |
1727 | + { | |
1728 | + switch (bytemode) | |
1729 | + { | |
1730 | + case b_mode: | |
1731 | + oappend (names8[rm]); | |
1732 | + break; | |
1733 | + case w_mode: | |
1734 | + oappend (names16[rm]); | |
1735 | + break; | |
1736 | + case v_mode: | |
1737 | + if (dflag) | |
1738 | + oappend (names32[rm]); | |
1739 | + else | |
1740 | + oappend (names16[rm]); | |
1741 | + break; | |
1742 | + default: | |
1743 | + oappend ("<bad dis table>"); | |
1744 | + break; | |
1745 | + } | |
1746 | + return 0; | |
1747 | + } | |
1748 | + | |
1749 | + disp = 0; | |
1750 | + append_prefix (); | |
1751 | + | |
1752 | + if (aflag) /* 32 bit address mode */ | |
1753 | + { | |
1754 | + int havesib; | |
1755 | + int havebase; | |
1756 | + int base; | |
1757 | + int index = 0; | |
1758 | + int scale = 0; | |
1759 | + | |
1760 | + havesib = 0; | |
1761 | + havebase = 1; | |
1762 | + base = rm; | |
1763 | + | |
1764 | + if (base == 4) | |
1765 | + { | |
1766 | + havesib = 1; | |
1767 | + FETCH_DATA (the_info, codep + 1); | |
1768 | + scale = (*codep >> 6) & 3; | |
1769 | + index = (*codep >> 3) & 7; | |
1770 | + base = *codep & 7; | |
1771 | + codep++; | |
1772 | + } | |
1773 | + | |
1774 | + switch (mod) | |
1775 | + { | |
1776 | + case 0: | |
1777 | + if (base == 5) | |
1778 | + { | |
1779 | + havebase = 0; | |
1780 | + disp = get32 (); | |
1781 | + } | |
1782 | + break; | |
1783 | + case 1: | |
1784 | + FETCH_DATA (the_info, codep + 1); | |
1785 | + disp = *codep++; | |
1786 | + if ((disp & 0x80) != 0) | |
1787 | + disp -= 0x100; | |
1788 | + break; | |
1789 | + case 2: | |
1790 | + disp = get32 (); | |
1791 | + break; | |
1792 | + } | |
1793 | + | |
1794 | + if (mod != 0 || base == 5) | |
1795 | + { | |
1796 | + sprintf (scratchbuf, "0x%x", disp); | |
1797 | + oappend (scratchbuf); | |
1798 | + } | |
1799 | + | |
1800 | + if (havebase || (havesib && (index != 4 || scale != 0))) | |
1801 | + { | |
1802 | + oappend ("("); | |
1803 | + if (havebase) | |
1804 | + oappend (names32[base]); | |
1805 | + if (havesib) | |
1806 | + { | |
1807 | + if (index != 4) | |
1808 | + { | |
1809 | + sprintf (scratchbuf, ",%s", names32[index]); | |
1810 | + oappend (scratchbuf); | |
1811 | + } | |
1812 | + sprintf (scratchbuf, ",%d", 1 << scale); | |
1813 | + oappend (scratchbuf); | |
1814 | + } | |
1815 | + oappend (")"); | |
1816 | + } | |
1817 | + } | |
1818 | + else | |
1819 | + { /* 16 bit address mode */ | |
1820 | + switch (mod) | |
1821 | + { | |
1822 | + case 0: | |
1823 | + if (rm == 6) | |
1824 | + { | |
1825 | + disp = get16 (); | |
1826 | + if ((disp & 0x8000) != 0) | |
1827 | + disp -= 0x10000; | |
1828 | + } | |
1829 | + break; | |
1830 | + case 1: | |
1831 | + FETCH_DATA (the_info, codep + 1); | |
1832 | + disp = *codep++; | |
1833 | + if ((disp & 0x80) != 0) | |
1834 | + disp -= 0x100; | |
1835 | + break; | |
1836 | + case 2: | |
1837 | + disp = get16 (); | |
1838 | + if ((disp & 0x8000) != 0) | |
1839 | + disp -= 0x10000; | |
1840 | + break; | |
1841 | + } | |
1842 | + | |
1843 | + if (mod != 0 || rm == 6) | |
1844 | + { | |
1845 | + sprintf (scratchbuf, "0x%x", disp); | |
1846 | + oappend (scratchbuf); | |
1847 | + } | |
1848 | + | |
1849 | + if (mod != 0 || rm != 6) | |
1850 | + { | |
1851 | + oappend ("("); | |
1852 | + oappend (index16[rm]); | |
1853 | + oappend (")"); | |
1854 | + } | |
1855 | + } | |
1856 | + return 0; | |
1857 | +} | |
1858 | + | |
1859 | +static int | |
1860 | +OP_G (bytemode, aflag, dflag) | |
1861 | + int bytemode; | |
1862 | + int aflag; | |
1863 | + int dflag; | |
1864 | +{ | |
1865 | + switch (bytemode) | |
1866 | + { | |
1867 | + case b_mode: | |
1868 | + oappend (names8[reg]); | |
1869 | + break; | |
1870 | + case w_mode: | |
1871 | + oappend (names16[reg]); | |
1872 | + break; | |
1873 | + case d_mode: | |
1874 | + oappend (names32[reg]); | |
1875 | + break; | |
1876 | + case v_mode: | |
1877 | + if (dflag) | |
1878 | + oappend (names32[reg]); | |
1879 | + else | |
1880 | + oappend (names16[reg]); | |
1881 | + break; | |
1882 | + default: | |
1883 | + oappend ("<internal disassembler error>"); | |
1884 | + break; | |
1885 | + } | |
1886 | + return (0); | |
1887 | +} | |
1888 | + | |
1889 | +static int | |
1890 | +get32 () | |
1891 | +{ | |
1892 | + int x = 0; | |
1893 | + | |
1894 | + FETCH_DATA (the_info, codep + 4); | |
1895 | + x = *codep++ & 0xff; | |
1896 | + x |= (*codep++ & 0xff) << 8; | |
1897 | + x |= (*codep++ & 0xff) << 16; | |
1898 | + x |= (*codep++ & 0xff) << 24; | |
1899 | + return (x); | |
1900 | +} | |
1901 | + | |
1902 | +static int | |
1903 | +get16 () | |
1904 | +{ | |
1905 | + int x = 0; | |
1906 | + | |
1907 | + FETCH_DATA (the_info, codep + 2); | |
1908 | + x = *codep++ & 0xff; | |
1909 | + x |= (*codep++ & 0xff) << 8; | |
1910 | + return (x); | |
1911 | +} | |
1912 | + | |
1913 | +static void | |
1914 | +set_op (op) | |
1915 | + int op; | |
1916 | +{ | |
1917 | + op_index[op_ad] = op_ad; | |
1918 | + op_address[op_ad] = op; | |
1919 | +} | |
1920 | + | |
1921 | +static int | |
1922 | +OP_REG (code, aflag, dflag) | |
1923 | + int code; | |
1924 | + int aflag; | |
1925 | + int dflag; | |
1926 | +{ | |
1927 | + char *s; | |
1928 | + | |
1929 | + switch (code) | |
1930 | + { | |
1931 | + case indir_dx_reg: s = "(%dx)"; break; | |
1932 | + case ax_reg: case cx_reg: case dx_reg: case bx_reg: | |
1933 | + case sp_reg: case bp_reg: case si_reg: case di_reg: | |
1934 | + s = names16[code - ax_reg]; | |
1935 | + break; | |
1936 | + case es_reg: case ss_reg: case cs_reg: | |
1937 | + case ds_reg: case fs_reg: case gs_reg: | |
1938 | + s = names_seg[code - es_reg]; | |
1939 | + break; | |
1940 | + case al_reg: case ah_reg: case cl_reg: case ch_reg: | |
1941 | + case dl_reg: case dh_reg: case bl_reg: case bh_reg: | |
1942 | + s = names8[code - al_reg]; | |
1943 | + break; | |
1944 | + case eAX_reg: case eCX_reg: case eDX_reg: case eBX_reg: | |
1945 | + case eSP_reg: case eBP_reg: case eSI_reg: case eDI_reg: | |
1946 | + if (dflag) | |
1947 | + s = names32[code - eAX_reg]; | |
1948 | + else | |
1949 | + s = names16[code - eAX_reg]; | |
1950 | + break; | |
1951 | + default: | |
1952 | + s = "<internal disassembler error>"; | |
1953 | + break; | |
1954 | + } | |
1955 | + oappend (s); | |
1956 | + return (0); | |
1957 | +} | |
1958 | + | |
1959 | +static int | |
1960 | +OP_I (bytemode, aflag, dflag) | |
1961 | + int bytemode; | |
1962 | + int aflag; | |
1963 | + int dflag; | |
1964 | +{ | |
1965 | + int op; | |
1966 | + | |
1967 | + switch (bytemode) | |
1968 | + { | |
1969 | + case b_mode: | |
1970 | + FETCH_DATA (the_info, codep + 1); | |
1971 | + op = *codep++ & 0xff; | |
1972 | + break; | |
1973 | + case v_mode: | |
1974 | + if (dflag) | |
1975 | + op = get32 (); | |
1976 | + else | |
1977 | + op = get16 (); | |
1978 | + break; | |
1979 | + case w_mode: | |
1980 | + op = get16 (); | |
1981 | + break; | |
1982 | + default: | |
1983 | + oappend ("<internal disassembler error>"); | |
1984 | + return (0); | |
1985 | + } | |
1986 | + sprintf (scratchbuf, "$0x%x", op); | |
1987 | + oappend (scratchbuf); | |
1988 | + return (0); | |
1989 | +} | |
1990 | + | |
1991 | +static int | |
1992 | +OP_sI (bytemode, aflag, dflag) | |
1993 | + int bytemode; | |
1994 | + int aflag; | |
1995 | + int dflag; | |
1996 | +{ | |
1997 | + int op; | |
1998 | + | |
1999 | + switch (bytemode) | |
2000 | + { | |
2001 | + case b_mode: | |
2002 | + FETCH_DATA (the_info, codep + 1); | |
2003 | + op = *codep++; | |
2004 | + if ((op & 0x80) != 0) | |
2005 | + op -= 0x100; | |
2006 | + break; | |
2007 | + case v_mode: | |
2008 | + if (dflag) | |
2009 | + op = get32 (); | |
2010 | + else | |
2011 | + { | |
2012 | + op = get16(); | |
2013 | + if ((op & 0x8000) != 0) | |
2014 | + op -= 0x10000; | |
2015 | + } | |
2016 | + break; | |
2017 | + case w_mode: | |
2018 | + op = get16 (); | |
2019 | + if ((op & 0x8000) != 0) | |
2020 | + op -= 0x10000; | |
2021 | + break; | |
2022 | + default: | |
2023 | + oappend ("<internal disassembler error>"); | |
2024 | + return (0); | |
2025 | + } | |
2026 | + sprintf (scratchbuf, "$0x%x", op); | |
2027 | + oappend (scratchbuf); | |
2028 | + return (0); | |
2029 | +} | |
2030 | + | |
2031 | +static int | |
2032 | +OP_J (bytemode, aflag, dflag) | |
2033 | + int bytemode; | |
2034 | + int aflag; | |
2035 | + int dflag; | |
2036 | +{ | |
2037 | + int disp; | |
2038 | + int mask = -1; | |
2039 | + | |
2040 | + switch (bytemode) | |
2041 | + { | |
2042 | + case b_mode: | |
2043 | + FETCH_DATA (the_info, codep + 1); | |
2044 | + disp = *codep++; | |
2045 | + if ((disp & 0x80) != 0) | |
2046 | + disp -= 0x100; | |
2047 | + break; | |
2048 | + case v_mode: | |
2049 | + if (dflag) | |
2050 | + disp = get32 (); | |
2051 | + else | |
2052 | + { | |
2053 | + disp = get16 (); | |
2054 | + if ((disp & 0x8000) != 0) | |
2055 | + disp -= 0x10000; | |
2056 | + /* for some reason, a data16 prefix on a jump instruction | |
2057 | + means that the pc is masked to 16 bits after the | |
2058 | + displacement is added! */ | |
2059 | + mask = 0xffff; | |
2060 | + } | |
2061 | + break; | |
2062 | + default: | |
2063 | + oappend ("<internal disassembler error>"); | |
2064 | + return (0); | |
2065 | + } | |
2066 | + disp = (start_pc + codep - start_codep + disp) & mask; | |
2067 | + set_op (disp); | |
2068 | + sprintf (scratchbuf, "0x%x", disp); | |
2069 | + oappend (scratchbuf); | |
2070 | + return (0); | |
2071 | +} | |
2072 | + | |
2073 | +/* ARGSUSED */ | |
2074 | +static int | |
2075 | +OP_SEG (dummy, aflag, dflag) | |
2076 | + int dummy; | |
2077 | + int aflag; | |
2078 | + int dflag; | |
2079 | +{ | |
2080 | + static char *sreg[] = { | |
2081 | + "%es","%cs","%ss","%ds","%fs","%gs","%?","%?", | |
2082 | + }; | |
2083 | + | |
2084 | + oappend (sreg[reg]); | |
2085 | + return (0); | |
2086 | +} | |
2087 | + | |
2088 | +static int | |
2089 | +OP_DIR (size, aflag, dflag) | |
2090 | + int size; | |
2091 | + int aflag; | |
2092 | + int dflag; | |
2093 | +{ | |
2094 | + int seg, offset; | |
2095 | + | |
2096 | + switch (size) | |
2097 | + { | |
2098 | + case lptr: | |
2099 | + if (aflag) | |
2100 | + { | |
2101 | + offset = get32 (); | |
2102 | + seg = get16 (); | |
2103 | + } | |
2104 | + else | |
2105 | + { | |
2106 | + offset = get16 (); | |
2107 | + seg = get16 (); | |
2108 | + } | |
2109 | + sprintf (scratchbuf, "0x%x,0x%x", seg, offset); | |
2110 | + oappend (scratchbuf); | |
2111 | + break; | |
2112 | + case v_mode: | |
2113 | + if (aflag) | |
2114 | + offset = get32 (); | |
2115 | + else | |
2116 | + { | |
2117 | + offset = get16 (); | |
2118 | + if ((offset & 0x8000) != 0) | |
2119 | + offset -= 0x10000; | |
2120 | + } | |
2121 | + | |
2122 | + offset = start_pc + codep - start_codep + offset; | |
2123 | + set_op (offset); | |
2124 | + sprintf (scratchbuf, "0x%x", offset); | |
2125 | + oappend (scratchbuf); | |
2126 | + break; | |
2127 | + default: | |
2128 | + oappend ("<internal disassembler error>"); | |
2129 | + break; | |
2130 | + } | |
2131 | + return (0); | |
2132 | +} | |
2133 | + | |
2134 | +/* ARGSUSED */ | |
2135 | +static int | |
2136 | +OP_OFF (bytemode, aflag, dflag) | |
2137 | + int bytemode; | |
2138 | + int aflag; | |
2139 | + int dflag; | |
2140 | +{ | |
2141 | + int off; | |
2142 | + | |
2143 | + append_prefix (); | |
2144 | + | |
2145 | + if (aflag) | |
2146 | + off = get32 (); | |
2147 | + else | |
2148 | + off = get16 (); | |
2149 | + | |
2150 | + sprintf (scratchbuf, "0x%x", off); | |
2151 | + oappend (scratchbuf); | |
2152 | + return (0); | |
2153 | +} | |
2154 | + | |
2155 | +/* ARGSUSED */ | |
2156 | +static int | |
2157 | +OP_ESDI (dummy, aflag, dflag) | |
2158 | + int dummy; | |
2159 | + int aflag; | |
2160 | + int dflag; | |
2161 | +{ | |
2162 | + oappend ("%es:("); | |
2163 | + oappend (aflag ? "%edi" : "%di"); | |
2164 | + oappend (")"); | |
2165 | + return (0); | |
2166 | +} | |
2167 | + | |
2168 | +/* ARGSUSED */ | |
2169 | +static int | |
2170 | +OP_DSSI (dummy, aflag, dflag) | |
2171 | + int dummy; | |
2172 | + int aflag; | |
2173 | + int dflag; | |
2174 | +{ | |
2175 | + if ((prefixes | |
2176 | + & (PREFIX_CS | |
2177 | + | PREFIX_DS | |
2178 | + | PREFIX_SS | |
2179 | + | PREFIX_ES | |
2180 | + | PREFIX_FS | |
2181 | + | PREFIX_GS)) == 0) | |
2182 | + prefixes |= PREFIX_DS; | |
2183 | + append_prefix (); | |
2184 | + oappend ("("); | |
2185 | + oappend (aflag ? "%esi" : "%si"); | |
2186 | + oappend (")"); | |
2187 | + return (0); | |
2188 | +} | |
2189 | + | |
2190 | +#if 0 | |
2191 | +/* Not used. */ | |
2192 | + | |
2193 | +/* ARGSUSED */ | |
2194 | +static int | |
2195 | +OP_ONE (dummy, aflag, dflag) | |
2196 | + int dummy; | |
2197 | + int aflag; | |
2198 | + int dflag; | |
2199 | +{ | |
2200 | + oappend ("1"); | |
2201 | + return (0); | |
2202 | +} | |
2203 | + | |
2204 | +#endif | |
2205 | + | |
2206 | +/* ARGSUSED */ | |
2207 | +static int | |
2208 | +OP_C (dummy, aflag, dflag) | |
2209 | + int dummy; | |
2210 | + int aflag; | |
2211 | + int dflag; | |
2212 | +{ | |
2213 | + codep++; /* skip mod/rm */ | |
2214 | + sprintf (scratchbuf, "%%cr%d", reg); | |
2215 | + oappend (scratchbuf); | |
2216 | + return (0); | |
2217 | +} | |
2218 | + | |
2219 | +/* ARGSUSED */ | |
2220 | +static int | |
2221 | +OP_D (dummy, aflag, dflag) | |
2222 | + int dummy; | |
2223 | + int aflag; | |
2224 | + int dflag; | |
2225 | +{ | |
2226 | + codep++; /* skip mod/rm */ | |
2227 | + sprintf (scratchbuf, "%%db%d", reg); | |
2228 | + oappend (scratchbuf); | |
2229 | + return (0); | |
2230 | +} | |
2231 | + | |
2232 | +/* ARGSUSED */ | |
2233 | +static int | |
2234 | +OP_T (dummy, aflag, dflag) | |
2235 | + int dummy; | |
2236 | + int aflag; | |
2237 | + int dflag; | |
2238 | +{ | |
2239 | + codep++; /* skip mod/rm */ | |
2240 | + sprintf (scratchbuf, "%%tr%d", reg); | |
2241 | + oappend (scratchbuf); | |
2242 | + return (0); | |
2243 | +} | |
2244 | + | |
2245 | +static int | |
2246 | +OP_rm (bytemode, aflag, dflag) | |
2247 | + int bytemode; | |
2248 | + int aflag; | |
2249 | + int dflag; | |
2250 | +{ | |
2251 | + switch (bytemode) | |
2252 | + { | |
2253 | + case d_mode: | |
2254 | + oappend (names32[rm]); | |
2255 | + break; | |
2256 | + case w_mode: | |
2257 | + oappend (names16[rm]); | |
2258 | + break; | |
2259 | + } | |
2260 | + return (0); | |
2261 | +} | |
2262 | + | |
2263 | +static int | |
2264 | +OP_MMX (bytemode, aflag, dflag) | |
2265 | + int bytemode; | |
2266 | + int aflag; | |
2267 | + int dflag; | |
2268 | +{ | |
2269 | + sprintf (scratchbuf, "%%mm%d", reg); | |
2270 | + oappend (scratchbuf); | |
2271 | + return 0; | |
2272 | +} | |
2273 | + | |
2274 | +static int | |
2275 | +OP_EM (bytemode, aflag, dflag) | |
2276 | + int bytemode; | |
2277 | + int aflag; | |
2278 | + int dflag; | |
2279 | +{ | |
2280 | + if (mod != 3) | |
2281 | + return OP_E (bytemode, aflag, dflag); | |
2282 | + | |
2283 | + codep++; | |
2284 | + sprintf (scratchbuf, "%%mm%d", rm); | |
2285 | + oappend (scratchbuf); | |
2286 | + return 0; | |
2287 | +} | |
2288 | + | |
2289 | +static int | |
2290 | +OP_MS (bytemode, aflag, dflag) | |
2291 | + int bytemode; | |
2292 | + int aflag; | |
2293 | + int dflag; | |
2294 | +{ | |
2295 | + ++codep; | |
2296 | + sprintf (scratchbuf, "%%mm%d", rm); | |
2297 | + oappend (scratchbuf); | |
2298 | + return 0; | |
2299 | +} | ... | ... |
linux-user/syscall.c
op-i386.c
... | ... | @@ -177,7 +177,7 @@ void raise_exception(int exception_index) |
177 | 177 | #undef REG |
178 | 178 | #undef REGNAME |
179 | 179 | |
180 | -/* operations */ | |
180 | +/* operations with flags */ | |
181 | 181 | |
182 | 182 | void OPPROTO op_addl_T0_T1_cc(void) |
183 | 183 | { |
... | ... | @@ -217,11 +217,6 @@ void OPPROTO op_cmpl_T0_T1_cc(void) |
217 | 217 | CC_DST = T0 - T1; |
218 | 218 | } |
219 | 219 | |
220 | -void OPPROTO op_notl_T0(void) | |
221 | -{ | |
222 | - T0 = ~T0; | |
223 | -} | |
224 | - | |
225 | 220 | void OPPROTO op_negl_T0_cc(void) |
226 | 221 | { |
227 | 222 | CC_SRC = 0; |
... | ... | @@ -248,6 +243,53 @@ void OPPROTO op_testl_T0_T1_cc(void) |
248 | 243 | CC_DST = T0 & T1; |
249 | 244 | } |
250 | 245 | |
246 | +/* operations without flags */ | |
247 | + | |
248 | +void OPPROTO op_addl_T0_T1(void) | |
249 | +{ | |
250 | + T0 += T1; | |
251 | +} | |
252 | + | |
253 | +void OPPROTO op_orl_T0_T1(void) | |
254 | +{ | |
255 | + T0 |= T1; | |
256 | +} | |
257 | + | |
258 | +void OPPROTO op_andl_T0_T1(void) | |
259 | +{ | |
260 | + T0 &= T1; | |
261 | +} | |
262 | + | |
263 | +void OPPROTO op_subl_T0_T1(void) | |
264 | +{ | |
265 | + T0 -= T1; | |
266 | +} | |
267 | + | |
268 | +void OPPROTO op_xorl_T0_T1(void) | |
269 | +{ | |
270 | + T0 ^= T1; | |
271 | +} | |
272 | + | |
273 | +void OPPROTO op_negl_T0(void) | |
274 | +{ | |
275 | + T0 = -T0; | |
276 | +} | |
277 | + | |
278 | +void OPPROTO op_incl_T0(void) | |
279 | +{ | |
280 | + T0++; | |
281 | +} | |
282 | + | |
283 | +void OPPROTO op_decl_T0(void) | |
284 | +{ | |
285 | + T0--; | |
286 | +} | |
287 | + | |
288 | +void OPPROTO op_notl_T0(void) | |
289 | +{ | |
290 | + T0 = ~T0; | |
291 | +} | |
292 | + | |
251 | 293 | void OPPROTO op_bswapl_T0(void) |
252 | 294 | { |
253 | 295 | T0 = bswap32(T0); | ... | ... |
opc-i386.h
0 → 100644
1 | +DEF(end) | |
2 | +DEF(movl_A0_EAX) | |
3 | +DEF(addl_A0_EAX) | |
4 | +DEF(addl_A0_EAX_s1) | |
5 | +DEF(addl_A0_EAX_s2) | |
6 | +DEF(addl_A0_EAX_s3) | |
7 | +DEF(movl_T0_EAX) | |
8 | +DEF(movl_T1_EAX) | |
9 | +DEF(movh_T0_EAX) | |
10 | +DEF(movh_T1_EAX) | |
11 | +DEF(movl_EAX_T0) | |
12 | +DEF(movl_EAX_T1) | |
13 | +DEF(movl_EAX_A0) | |
14 | +DEF(cmovw_EAX_T1_T0) | |
15 | +DEF(cmovl_EAX_T1_T0) | |
16 | +DEF(movw_EAX_T0) | |
17 | +DEF(movw_EAX_T1) | |
18 | +DEF(movw_EAX_A0) | |
19 | +DEF(movb_EAX_T0) | |
20 | +DEF(movh_EAX_T0) | |
21 | +DEF(movb_EAX_T1) | |
22 | +DEF(movh_EAX_T1) | |
23 | +DEF(movl_A0_ECX) | |
24 | +DEF(addl_A0_ECX) | |
25 | +DEF(addl_A0_ECX_s1) | |
26 | +DEF(addl_A0_ECX_s2) | |
27 | +DEF(addl_A0_ECX_s3) | |
28 | +DEF(movl_T0_ECX) | |
29 | +DEF(movl_T1_ECX) | |
30 | +DEF(movh_T0_ECX) | |
31 | +DEF(movh_T1_ECX) | |
32 | +DEF(movl_ECX_T0) | |
33 | +DEF(movl_ECX_T1) | |
34 | +DEF(movl_ECX_A0) | |
35 | +DEF(cmovw_ECX_T1_T0) | |
36 | +DEF(cmovl_ECX_T1_T0) | |
37 | +DEF(movw_ECX_T0) | |
38 | +DEF(movw_ECX_T1) | |
39 | +DEF(movw_ECX_A0) | |
40 | +DEF(movb_ECX_T0) | |
41 | +DEF(movh_ECX_T0) | |
42 | +DEF(movb_ECX_T1) | |
43 | +DEF(movh_ECX_T1) | |
44 | +DEF(movl_A0_EDX) | |
45 | +DEF(addl_A0_EDX) | |
46 | +DEF(addl_A0_EDX_s1) | |
47 | +DEF(addl_A0_EDX_s2) | |
48 | +DEF(addl_A0_EDX_s3) | |
49 | +DEF(movl_T0_EDX) | |
50 | +DEF(movl_T1_EDX) | |
51 | +DEF(movh_T0_EDX) | |
52 | +DEF(movh_T1_EDX) | |
53 | +DEF(movl_EDX_T0) | |
54 | +DEF(movl_EDX_T1) | |
55 | +DEF(movl_EDX_A0) | |
56 | +DEF(cmovw_EDX_T1_T0) | |
57 | +DEF(cmovl_EDX_T1_T0) | |
58 | +DEF(movw_EDX_T0) | |
59 | +DEF(movw_EDX_T1) | |
60 | +DEF(movw_EDX_A0) | |
61 | +DEF(movb_EDX_T0) | |
62 | +DEF(movh_EDX_T0) | |
63 | +DEF(movb_EDX_T1) | |
64 | +DEF(movh_EDX_T1) | |
65 | +DEF(movl_A0_EBX) | |
66 | +DEF(addl_A0_EBX) | |
67 | +DEF(addl_A0_EBX_s1) | |
68 | +DEF(addl_A0_EBX_s2) | |
69 | +DEF(addl_A0_EBX_s3) | |
70 | +DEF(movl_T0_EBX) | |
71 | +DEF(movl_T1_EBX) | |
72 | +DEF(movh_T0_EBX) | |
73 | +DEF(movh_T1_EBX) | |
74 | +DEF(movl_EBX_T0) | |
75 | +DEF(movl_EBX_T1) | |
76 | +DEF(movl_EBX_A0) | |
77 | +DEF(cmovw_EBX_T1_T0) | |
78 | +DEF(cmovl_EBX_T1_T0) | |
79 | +DEF(movw_EBX_T0) | |
80 | +DEF(movw_EBX_T1) | |
81 | +DEF(movw_EBX_A0) | |
82 | +DEF(movb_EBX_T0) | |
83 | +DEF(movh_EBX_T0) | |
84 | +DEF(movb_EBX_T1) | |
85 | +DEF(movh_EBX_T1) | |
86 | +DEF(movl_A0_ESP) | |
87 | +DEF(addl_A0_ESP) | |
88 | +DEF(addl_A0_ESP_s1) | |
89 | +DEF(addl_A0_ESP_s2) | |
90 | +DEF(addl_A0_ESP_s3) | |
91 | +DEF(movl_T0_ESP) | |
92 | +DEF(movl_T1_ESP) | |
93 | +DEF(movh_T0_ESP) | |
94 | +DEF(movh_T1_ESP) | |
95 | +DEF(movl_ESP_T0) | |
96 | +DEF(movl_ESP_T1) | |
97 | +DEF(movl_ESP_A0) | |
98 | +DEF(cmovw_ESP_T1_T0) | |
99 | +DEF(cmovl_ESP_T1_T0) | |
100 | +DEF(movw_ESP_T0) | |
101 | +DEF(movw_ESP_T1) | |
102 | +DEF(movw_ESP_A0) | |
103 | +DEF(movb_ESP_T0) | |
104 | +DEF(movh_ESP_T0) | |
105 | +DEF(movb_ESP_T1) | |
106 | +DEF(movh_ESP_T1) | |
107 | +DEF(movl_A0_EBP) | |
108 | +DEF(addl_A0_EBP) | |
109 | +DEF(addl_A0_EBP_s1) | |
110 | +DEF(addl_A0_EBP_s2) | |
111 | +DEF(addl_A0_EBP_s3) | |
112 | +DEF(movl_T0_EBP) | |
113 | +DEF(movl_T1_EBP) | |
114 | +DEF(movh_T0_EBP) | |
115 | +DEF(movh_T1_EBP) | |
116 | +DEF(movl_EBP_T0) | |
117 | +DEF(movl_EBP_T1) | |
118 | +DEF(movl_EBP_A0) | |
119 | +DEF(cmovw_EBP_T1_T0) | |
120 | +DEF(cmovl_EBP_T1_T0) | |
121 | +DEF(movw_EBP_T0) | |
122 | +DEF(movw_EBP_T1) | |
123 | +DEF(movw_EBP_A0) | |
124 | +DEF(movb_EBP_T0) | |
125 | +DEF(movh_EBP_T0) | |
126 | +DEF(movb_EBP_T1) | |
127 | +DEF(movh_EBP_T1) | |
128 | +DEF(movl_A0_ESI) | |
129 | +DEF(addl_A0_ESI) | |
130 | +DEF(addl_A0_ESI_s1) | |
131 | +DEF(addl_A0_ESI_s2) | |
132 | +DEF(addl_A0_ESI_s3) | |
133 | +DEF(movl_T0_ESI) | |
134 | +DEF(movl_T1_ESI) | |
135 | +DEF(movh_T0_ESI) | |
136 | +DEF(movh_T1_ESI) | |
137 | +DEF(movl_ESI_T0) | |
138 | +DEF(movl_ESI_T1) | |
139 | +DEF(movl_ESI_A0) | |
140 | +DEF(cmovw_ESI_T1_T0) | |
141 | +DEF(cmovl_ESI_T1_T0) | |
142 | +DEF(movw_ESI_T0) | |
143 | +DEF(movw_ESI_T1) | |
144 | +DEF(movw_ESI_A0) | |
145 | +DEF(movb_ESI_T0) | |
146 | +DEF(movh_ESI_T0) | |
147 | +DEF(movb_ESI_T1) | |
148 | +DEF(movh_ESI_T1) | |
149 | +DEF(movl_A0_EDI) | |
150 | +DEF(addl_A0_EDI) | |
151 | +DEF(addl_A0_EDI_s1) | |
152 | +DEF(addl_A0_EDI_s2) | |
153 | +DEF(addl_A0_EDI_s3) | |
154 | +DEF(movl_T0_EDI) | |
155 | +DEF(movl_T1_EDI) | |
156 | +DEF(movh_T0_EDI) | |
157 | +DEF(movh_T1_EDI) | |
158 | +DEF(movl_EDI_T0) | |
159 | +DEF(movl_EDI_T1) | |
160 | +DEF(movl_EDI_A0) | |
161 | +DEF(cmovw_EDI_T1_T0) | |
162 | +DEF(cmovl_EDI_T1_T0) | |
163 | +DEF(movw_EDI_T0) | |
164 | +DEF(movw_EDI_T1) | |
165 | +DEF(movw_EDI_A0) | |
166 | +DEF(movb_EDI_T0) | |
167 | +DEF(movh_EDI_T0) | |
168 | +DEF(movb_EDI_T1) | |
169 | +DEF(movh_EDI_T1) | |
170 | +DEF(addl_T0_T1_cc) | |
171 | +DEF(orl_T0_T1_cc) | |
172 | +DEF(andl_T0_T1_cc) | |
173 | +DEF(subl_T0_T1_cc) | |
174 | +DEF(xorl_T0_T1_cc) | |
175 | +DEF(cmpl_T0_T1_cc) | |
176 | +DEF(negl_T0_cc) | |
177 | +DEF(incl_T0_cc) | |
178 | +DEF(decl_T0_cc) | |
179 | +DEF(testl_T0_T1_cc) | |
180 | +DEF(addl_T0_T1) | |
181 | +DEF(orl_T0_T1) | |
182 | +DEF(andl_T0_T1) | |
183 | +DEF(subl_T0_T1) | |
184 | +DEF(xorl_T0_T1) | |
185 | +DEF(negl_T0) | |
186 | +DEF(incl_T0) | |
187 | +DEF(decl_T0) | |
188 | +DEF(notl_T0) | |
189 | +DEF(bswapl_T0) | |
190 | +DEF(mulb_AL_T0) | |
191 | +DEF(imulb_AL_T0) | |
192 | +DEF(mulw_AX_T0) | |
193 | +DEF(imulw_AX_T0) | |
194 | +DEF(mull_EAX_T0) | |
195 | +DEF(imull_EAX_T0) | |
196 | +DEF(imulw_T0_T1) | |
197 | +DEF(imull_T0_T1) | |
198 | +DEF(divb_AL_T0) | |
199 | +DEF(idivb_AL_T0) | |
200 | +DEF(divw_AX_T0) | |
201 | +DEF(idivw_AX_T0) | |
202 | +DEF(divl_EAX_T0) | |
203 | +DEF(idivl_EAX_T0) | |
204 | +DEF(movl_T0_im) | |
205 | +DEF(movl_T1_im) | |
206 | +DEF(movl_A0_im) | |
207 | +DEF(addl_A0_im) | |
208 | +DEF(andl_A0_ffff) | |
209 | +DEF(ldub_T0_A0) | |
210 | +DEF(ldsb_T0_A0) | |
211 | +DEF(lduw_T0_A0) | |
212 | +DEF(ldsw_T0_A0) | |
213 | +DEF(ldl_T0_A0) | |
214 | +DEF(ldub_T1_A0) | |
215 | +DEF(ldsb_T1_A0) | |
216 | +DEF(lduw_T1_A0) | |
217 | +DEF(ldsw_T1_A0) | |
218 | +DEF(ldl_T1_A0) | |
219 | +DEF(stb_T0_A0) | |
220 | +DEF(stw_T0_A0) | |
221 | +DEF(stl_T0_A0) | |
222 | +DEF(add_bitw_A0_T1) | |
223 | +DEF(add_bitl_A0_T1) | |
224 | +DEF(jmp_T0) | |
225 | +DEF(jmp_im) | |
226 | +DEF(int_im) | |
227 | +DEF(int3) | |
228 | +DEF(into) | |
229 | +DEF(jb_subb) | |
230 | +DEF(jz_subb) | |
231 | +DEF(jbe_subb) | |
232 | +DEF(js_subb) | |
233 | +DEF(jl_subb) | |
234 | +DEF(jle_subb) | |
235 | +DEF(setb_T0_subb) | |
236 | +DEF(setz_T0_subb) | |
237 | +DEF(setbe_T0_subb) | |
238 | +DEF(sets_T0_subb) | |
239 | +DEF(setl_T0_subb) | |
240 | +DEF(setle_T0_subb) | |
241 | +DEF(rolb_T0_T1_cc) | |
242 | +DEF(rolb_T0_T1) | |
243 | +DEF(rorb_T0_T1_cc) | |
244 | +DEF(rorb_T0_T1) | |
245 | +DEF(rclb_T0_T1_cc) | |
246 | +DEF(rcrb_T0_T1_cc) | |
247 | +DEF(shlb_T0_T1_cc) | |
248 | +DEF(shlb_T0_T1) | |
249 | +DEF(shrb_T0_T1_cc) | |
250 | +DEF(shrb_T0_T1) | |
251 | +DEF(sarb_T0_T1_cc) | |
252 | +DEF(sarb_T0_T1) | |
253 | +DEF(adcb_T0_T1_cc) | |
254 | +DEF(sbbb_T0_T1_cc) | |
255 | +DEF(cmpxchgb_T0_T1_EAX_cc) | |
256 | +DEF(movsb) | |
257 | +DEF(rep_movsb) | |
258 | +DEF(stosb) | |
259 | +DEF(rep_stosb) | |
260 | +DEF(lodsb) | |
261 | +DEF(rep_lodsb) | |
262 | +DEF(scasb) | |
263 | +DEF(repz_scasb) | |
264 | +DEF(repnz_scasb) | |
265 | +DEF(cmpsb) | |
266 | +DEF(repz_cmpsb) | |
267 | +DEF(repnz_cmpsb) | |
268 | +DEF(outsb) | |
269 | +DEF(rep_outsb) | |
270 | +DEF(insb) | |
271 | +DEF(rep_insb) | |
272 | +DEF(outb_T0_T1) | |
273 | +DEF(inb_T0_T1) | |
274 | +DEF(jb_subw) | |
275 | +DEF(jz_subw) | |
276 | +DEF(jbe_subw) | |
277 | +DEF(js_subw) | |
278 | +DEF(jl_subw) | |
279 | +DEF(jle_subw) | |
280 | +DEF(loopnzw) | |
281 | +DEF(loopzw) | |
282 | +DEF(loopw) | |
283 | +DEF(jecxzw) | |
284 | +DEF(setb_T0_subw) | |
285 | +DEF(setz_T0_subw) | |
286 | +DEF(setbe_T0_subw) | |
287 | +DEF(sets_T0_subw) | |
288 | +DEF(setl_T0_subw) | |
289 | +DEF(setle_T0_subw) | |
290 | +DEF(rolw_T0_T1_cc) | |
291 | +DEF(rolw_T0_T1) | |
292 | +DEF(rorw_T0_T1_cc) | |
293 | +DEF(rorw_T0_T1) | |
294 | +DEF(rclw_T0_T1_cc) | |
295 | +DEF(rcrw_T0_T1_cc) | |
296 | +DEF(shlw_T0_T1_cc) | |
297 | +DEF(shlw_T0_T1) | |
298 | +DEF(shrw_T0_T1_cc) | |
299 | +DEF(shrw_T0_T1) | |
300 | +DEF(sarw_T0_T1_cc) | |
301 | +DEF(sarw_T0_T1) | |
302 | +DEF(shldw_T0_T1_im_cc) | |
303 | +DEF(shldw_T0_T1_ECX_cc) | |
304 | +DEF(shrdw_T0_T1_im_cc) | |
305 | +DEF(shrdw_T0_T1_ECX_cc) | |
306 | +DEF(adcw_T0_T1_cc) | |
307 | +DEF(sbbw_T0_T1_cc) | |
308 | +DEF(cmpxchgw_T0_T1_EAX_cc) | |
309 | +DEF(btw_T0_T1_cc) | |
310 | +DEF(btsw_T0_T1_cc) | |
311 | +DEF(btrw_T0_T1_cc) | |
312 | +DEF(btcw_T0_T1_cc) | |
313 | +DEF(bsfw_T0_cc) | |
314 | +DEF(bsrw_T0_cc) | |
315 | +DEF(movsw) | |
316 | +DEF(rep_movsw) | |
317 | +DEF(stosw) | |
318 | +DEF(rep_stosw) | |
319 | +DEF(lodsw) | |
320 | +DEF(rep_lodsw) | |
321 | +DEF(scasw) | |
322 | +DEF(repz_scasw) | |
323 | +DEF(repnz_scasw) | |
324 | +DEF(cmpsw) | |
325 | +DEF(repz_cmpsw) | |
326 | +DEF(repnz_cmpsw) | |
327 | +DEF(outsw) | |
328 | +DEF(rep_outsw) | |
329 | +DEF(insw) | |
330 | +DEF(rep_insw) | |
331 | +DEF(outw_T0_T1) | |
332 | +DEF(inw_T0_T1) | |
333 | +DEF(jb_subl) | |
334 | +DEF(jz_subl) | |
335 | +DEF(jbe_subl) | |
336 | +DEF(js_subl) | |
337 | +DEF(jl_subl) | |
338 | +DEF(jle_subl) | |
339 | +DEF(loopnzl) | |
340 | +DEF(loopzl) | |
341 | +DEF(loopl) | |
342 | +DEF(jecxzl) | |
343 | +DEF(setb_T0_subl) | |
344 | +DEF(setz_T0_subl) | |
345 | +DEF(setbe_T0_subl) | |
346 | +DEF(sets_T0_subl) | |
347 | +DEF(setl_T0_subl) | |
348 | +DEF(setle_T0_subl) | |
349 | +DEF(roll_T0_T1_cc) | |
350 | +DEF(roll_T0_T1) | |
351 | +DEF(rorl_T0_T1_cc) | |
352 | +DEF(rorl_T0_T1) | |
353 | +DEF(rcll_T0_T1_cc) | |
354 | +DEF(rcrl_T0_T1_cc) | |
355 | +DEF(shll_T0_T1_cc) | |
356 | +DEF(shll_T0_T1) | |
357 | +DEF(shrl_T0_T1_cc) | |
358 | +DEF(shrl_T0_T1) | |
359 | +DEF(sarl_T0_T1_cc) | |
360 | +DEF(sarl_T0_T1) | |
361 | +DEF(shldl_T0_T1_im_cc) | |
362 | +DEF(shldl_T0_T1_ECX_cc) | |
363 | +DEF(shrdl_T0_T1_im_cc) | |
364 | +DEF(shrdl_T0_T1_ECX_cc) | |
365 | +DEF(adcl_T0_T1_cc) | |
366 | +DEF(sbbl_T0_T1_cc) | |
367 | +DEF(cmpxchgl_T0_T1_EAX_cc) | |
368 | +DEF(btl_T0_T1_cc) | |
369 | +DEF(btsl_T0_T1_cc) | |
370 | +DEF(btrl_T0_T1_cc) | |
371 | +DEF(btcl_T0_T1_cc) | |
372 | +DEF(bsfl_T0_cc) | |
373 | +DEF(bsrl_T0_cc) | |
374 | +DEF(movsl) | |
375 | +DEF(rep_movsl) | |
376 | +DEF(stosl) | |
377 | +DEF(rep_stosl) | |
378 | +DEF(lodsl) | |
379 | +DEF(rep_lodsl) | |
380 | +DEF(scasl) | |
381 | +DEF(repz_scasl) | |
382 | +DEF(repnz_scasl) | |
383 | +DEF(cmpsl) | |
384 | +DEF(repz_cmpsl) | |
385 | +DEF(repnz_cmpsl) | |
386 | +DEF(outsl) | |
387 | +DEF(rep_outsl) | |
388 | +DEF(insl) | |
389 | +DEF(rep_insl) | |
390 | +DEF(outl_T0_T1) | |
391 | +DEF(inl_T0_T1) | |
392 | +DEF(movsbl_T0_T0) | |
393 | +DEF(movzbl_T0_T0) | |
394 | +DEF(movswl_T0_T0) | |
395 | +DEF(movzwl_T0_T0) | |
396 | +DEF(movswl_EAX_AX) | |
397 | +DEF(movsbw_AX_AL) | |
398 | +DEF(movslq_EDX_EAX) | |
399 | +DEF(movswl_DX_AX) | |
400 | +DEF(pushl_T0) | |
401 | +DEF(pushl_T1) | |
402 | +DEF(popl_T0) | |
403 | +DEF(addl_ESP_im) | |
404 | +DEF(pushal) | |
405 | +DEF(pushaw) | |
406 | +DEF(popal) | |
407 | +DEF(popaw) | |
408 | +DEF(enterl) | |
409 | +DEF(rdtsc) | |
410 | +DEF(aam) | |
411 | +DEF(aad) | |
412 | +DEF(aaa) | |
413 | +DEF(aas) | |
414 | +DEF(daa) | |
415 | +DEF(das) | |
416 | +DEF(movl_seg_T0) | |
417 | +DEF(movl_T0_seg) | |
418 | +DEF(addl_A0_seg) | |
419 | +DEF(jo_cc) | |
420 | +DEF(jb_cc) | |
421 | +DEF(jz_cc) | |
422 | +DEF(jbe_cc) | |
423 | +DEF(js_cc) | |
424 | +DEF(jp_cc) | |
425 | +DEF(jl_cc) | |
426 | +DEF(jle_cc) | |
427 | +DEF(seto_T0_cc) | |
428 | +DEF(setb_T0_cc) | |
429 | +DEF(setz_T0_cc) | |
430 | +DEF(setbe_T0_cc) | |
431 | +DEF(sets_T0_cc) | |
432 | +DEF(setp_T0_cc) | |
433 | +DEF(setl_T0_cc) | |
434 | +DEF(setle_T0_cc) | |
435 | +DEF(xor_T0_1) | |
436 | +DEF(set_cc_op) | |
437 | +DEF(movl_eflags_T0) | |
438 | +DEF(movb_eflags_T0) | |
439 | +DEF(movl_T0_eflags) | |
440 | +DEF(cld) | |
441 | +DEF(std) | |
442 | +DEF(clc) | |
443 | +DEF(stc) | |
444 | +DEF(cmc) | |
445 | +DEF(salc) | |
446 | +DEF(flds_FT0_A0) | |
447 | +DEF(fldl_FT0_A0) | |
448 | +DEF(fild_FT0_A0) | |
449 | +DEF(fildl_FT0_A0) | |
450 | +DEF(fildll_FT0_A0) | |
451 | +DEF(flds_ST0_A0) | |
452 | +DEF(fldl_ST0_A0) | |
453 | +DEF(fldt_ST0_A0) | |
454 | +DEF(fild_ST0_A0) | |
455 | +DEF(fildl_ST0_A0) | |
456 | +DEF(fildll_ST0_A0) | |
457 | +DEF(fsts_ST0_A0) | |
458 | +DEF(fstl_ST0_A0) | |
459 | +DEF(fstt_ST0_A0) | |
460 | +DEF(fist_ST0_A0) | |
461 | +DEF(fistl_ST0_A0) | |
462 | +DEF(fistll_ST0_A0) | |
463 | +DEF(fbld_ST0_A0) | |
464 | +DEF(fbst_ST0_A0) | |
465 | +DEF(fpush) | |
466 | +DEF(fpop) | |
467 | +DEF(fdecstp) | |
468 | +DEF(fincstp) | |
469 | +DEF(fmov_ST0_FT0) | |
470 | +DEF(fmov_FT0_STN) | |
471 | +DEF(fmov_ST0_STN) | |
472 | +DEF(fmov_STN_ST0) | |
473 | +DEF(fxchg_ST0_STN) | |
474 | +DEF(fcom_ST0_FT0) | |
475 | +DEF(fucom_ST0_FT0) | |
476 | +DEF(fadd_ST0_FT0) | |
477 | +DEF(fmul_ST0_FT0) | |
478 | +DEF(fsub_ST0_FT0) | |
479 | +DEF(fsubr_ST0_FT0) | |
480 | +DEF(fdiv_ST0_FT0) | |
481 | +DEF(fdivr_ST0_FT0) | |
482 | +DEF(fadd_STN_ST0) | |
483 | +DEF(fmul_STN_ST0) | |
484 | +DEF(fsub_STN_ST0) | |
485 | +DEF(fsubr_STN_ST0) | |
486 | +DEF(fdiv_STN_ST0) | |
487 | +DEF(fdivr_STN_ST0) | |
488 | +DEF(fchs_ST0) | |
489 | +DEF(fabs_ST0) | |
490 | +DEF(fxam_ST0) | |
491 | +DEF(fld1_ST0) | |
492 | +DEF(fldl2t_ST0) | |
493 | +DEF(fldl2e_ST0) | |
494 | +DEF(fldpi_ST0) | |
495 | +DEF(fldlg2_ST0) | |
496 | +DEF(fldln2_ST0) | |
497 | +DEF(fldz_ST0) | |
498 | +DEF(fldz_FT0) | |
499 | +DEF(f2xm1) | |
500 | +DEF(fyl2x) | |
501 | +DEF(fptan) | |
502 | +DEF(fpatan) | |
503 | +DEF(fxtract) | |
504 | +DEF(fprem1) | |
505 | +DEF(fprem) | |
506 | +DEF(fyl2xp1) | |
507 | +DEF(fsqrt) | |
508 | +DEF(fsincos) | |
509 | +DEF(frndint) | |
510 | +DEF(fscale) | |
511 | +DEF(fsin) | |
512 | +DEF(fcos) | |
513 | +DEF(fnstsw_A0) | |
514 | +DEF(fnstsw_EAX) | |
515 | +DEF(fnstcw_A0) | |
516 | +DEF(fldcw_A0) | |
517 | +DEF(fclex) | |
518 | +DEF(fninit) | ... | ... |
ops_template.h
... | ... | @@ -385,7 +385,6 @@ void OPPROTO glue(op_setle_T0_sub, SUFFIX)(void) |
385 | 385 | void OPPROTO glue(glue(op_rol, SUFFIX), _T0_T1_cc)(void) |
386 | 386 | { |
387 | 387 | int count, src; |
388 | - /* XXX: testing */ | |
389 | 388 | count = T1 & SHIFT_MASK; |
390 | 389 | if (count) { |
391 | 390 | CC_SRC = cc_table[CC_OP].compute_all() & ~(CC_O | CC_C); |
... | ... | @@ -399,6 +398,17 @@ void OPPROTO glue(glue(op_rol, SUFFIX), _T0_T1_cc)(void) |
399 | 398 | FORCE_RET(); |
400 | 399 | } |
401 | 400 | |
401 | +void OPPROTO glue(glue(op_rol, SUFFIX), _T0_T1)(void) | |
402 | +{ | |
403 | + int count; | |
404 | + count = T1 & SHIFT_MASK; | |
405 | + if (count) { | |
406 | + T0 &= DATA_MASK; | |
407 | + T0 = (T0 << count) | (T0 >> (DATA_BITS - count)); | |
408 | + } | |
409 | + FORCE_RET(); | |
410 | +} | |
411 | + | |
402 | 412 | void OPPROTO glue(glue(op_ror, SUFFIX), _T0_T1_cc)(void) |
403 | 413 | { |
404 | 414 | int count, src; |
... | ... | @@ -415,6 +425,17 @@ void OPPROTO glue(glue(op_ror, SUFFIX), _T0_T1_cc)(void) |
415 | 425 | FORCE_RET(); |
416 | 426 | } |
417 | 427 | |
428 | +void OPPROTO glue(glue(op_ror, SUFFIX), _T0_T1)(void) | |
429 | +{ | |
430 | + int count; | |
431 | + count = T1 & SHIFT_MASK; | |
432 | + if (count) { | |
433 | + T0 &= DATA_MASK; | |
434 | + T0 = (T0 >> count) | (T0 << (DATA_BITS - count)); | |
435 | + } | |
436 | + FORCE_RET(); | |
437 | +} | |
438 | + | |
418 | 439 | void OPPROTO glue(glue(op_rcl, SUFFIX), _T0_T1_cc)(void) |
419 | 440 | { |
420 | 441 | int count, res, eflags; |
... | ... | @@ -482,6 +503,14 @@ void OPPROTO glue(glue(op_shl, SUFFIX), _T0_T1_cc)(void) |
482 | 503 | FORCE_RET(); |
483 | 504 | } |
484 | 505 | |
506 | +void OPPROTO glue(glue(op_shl, SUFFIX), _T0_T1)(void) | |
507 | +{ | |
508 | + int count; | |
509 | + count = T1 & 0x1f; | |
510 | + T0 = T0 << count; | |
511 | + FORCE_RET(); | |
512 | +} | |
513 | + | |
485 | 514 | void OPPROTO glue(glue(op_shr, SUFFIX), _T0_T1_cc)(void) |
486 | 515 | { |
487 | 516 | int count; |
... | ... | @@ -496,6 +525,15 @@ void OPPROTO glue(glue(op_shr, SUFFIX), _T0_T1_cc)(void) |
496 | 525 | FORCE_RET(); |
497 | 526 | } |
498 | 527 | |
528 | +void OPPROTO glue(glue(op_shr, SUFFIX), _T0_T1)(void) | |
529 | +{ | |
530 | + int count; | |
531 | + count = T1 & 0x1f; | |
532 | + T0 &= DATA_MASK; | |
533 | + T0 = T0 >> count; | |
534 | + FORCE_RET(); | |
535 | +} | |
536 | + | |
499 | 537 | void OPPROTO glue(glue(op_sar, SUFFIX), _T0_T1_cc)(void) |
500 | 538 | { |
501 | 539 | int count, src; |
... | ... | @@ -510,6 +548,15 @@ void OPPROTO glue(glue(op_sar, SUFFIX), _T0_T1_cc)(void) |
510 | 548 | FORCE_RET(); |
511 | 549 | } |
512 | 550 | |
551 | +void OPPROTO glue(glue(op_sar, SUFFIX), _T0_T1)(void) | |
552 | +{ | |
553 | + int count, src; | |
554 | + count = T1 & 0x1f; | |
555 | + src = (DATA_STYPE)T0; | |
556 | + T0 = src >> count; | |
557 | + FORCE_RET(); | |
558 | +} | |
559 | + | |
513 | 560 | #if DATA_BITS == 16 |
514 | 561 | /* XXX: overflow flag might be incorrect in some cases in shldw */ |
515 | 562 | void OPPROTO glue(glue(op_shld, SUFFIX), _T0_T1_im_cc)(void) | ... | ... |
translate-i386.c
... | ... | @@ -38,7 +38,8 @@ |
38 | 38 | #define offsetof(type, field) ((size_t) &((type *)0)->field) |
39 | 39 | #endif |
40 | 40 | |
41 | -static uint8_t *gen_code_ptr; | |
41 | +static uint16_t *gen_opc_ptr; | |
42 | +static uint32_t *gen_opparam_ptr; | |
42 | 43 | int __op_param1, __op_param2, __op_param3; |
43 | 44 | |
44 | 45 | extern FILE *logfile; |
... | ... | @@ -95,6 +96,13 @@ enum { |
95 | 96 | OP_SAR = 7, |
96 | 97 | }; |
97 | 98 | |
99 | +enum { | |
100 | +#define DEF(s) INDEX_op_ ## s, | |
101 | +#include "opc-i386.h" | |
102 | +#undef DEF | |
103 | + NB_OPS, | |
104 | +}; | |
105 | + | |
98 | 106 | #include "op-i386.h" |
99 | 107 | |
100 | 108 | /* operand size */ |
... | ... | @@ -1922,7 +1930,7 @@ long disas_insn(DisasContext *s, uint8_t *pc_start) |
1922 | 1930 | if (mod == 3) |
1923 | 1931 | goto illegal_op; |
1924 | 1932 | gen_op_ld_T1_A0[ot](); |
1925 | - op_addl_A0_im(1 << (ot - OT_WORD + 1)); | |
1933 | + gen_op_addl_A0_im(1 << (ot - OT_WORD + 1)); | |
1926 | 1934 | /* load the segment first to handle exceptions properly */ |
1927 | 1935 | gen_op_lduw_T0_A0(); |
1928 | 1936 | gen_movl_seg_T0(s, op); |
... | ... | @@ -2842,24 +2850,350 @@ long disas_insn(DisasContext *s, uint8_t *pc_start) |
2842 | 2850 | return -1; |
2843 | 2851 | } |
2844 | 2852 | |
2853 | +#define CC_OSZAPC (CC_O | CC_S | CC_Z | CC_A | CC_P | CC_C) | |
2854 | +#define CC_OSZAP (CC_O | CC_S | CC_Z | CC_A | CC_P) | |
2855 | + | |
2856 | +/* flags read by an operation */ | |
2857 | +static uint16_t opc_read_flags[NB_OPS] = { | |
2858 | + [INDEX_op_aas] = CC_A, | |
2859 | + [INDEX_op_aaa] = CC_A, | |
2860 | + [INDEX_op_das] = CC_A | CC_C, | |
2861 | + [INDEX_op_daa] = CC_A | CC_C, | |
2862 | + | |
2863 | + [INDEX_op_adcb_T0_T1_cc] = CC_C, | |
2864 | + [INDEX_op_adcw_T0_T1_cc] = CC_C, | |
2865 | + [INDEX_op_adcl_T0_T1_cc] = CC_C, | |
2866 | + [INDEX_op_sbbb_T0_T1_cc] = CC_C, | |
2867 | + [INDEX_op_sbbw_T0_T1_cc] = CC_C, | |
2868 | + [INDEX_op_sbbl_T0_T1_cc] = CC_C, | |
2869 | + | |
2870 | + [INDEX_op_into] = CC_O, | |
2871 | + | |
2872 | + [INDEX_op_jo_cc] = CC_O, | |
2873 | + [INDEX_op_jb_cc] = CC_C, | |
2874 | + [INDEX_op_jz_cc] = CC_Z, | |
2875 | + [INDEX_op_jbe_cc] = CC_Z | CC_C, | |
2876 | + [INDEX_op_js_cc] = CC_S, | |
2877 | + [INDEX_op_jp_cc] = CC_P, | |
2878 | + [INDEX_op_jl_cc] = CC_O | CC_S, | |
2879 | + [INDEX_op_jle_cc] = CC_O | CC_S | CC_Z, | |
2880 | + | |
2881 | + [INDEX_op_jb_subb] = CC_C, | |
2882 | + [INDEX_op_jb_subw] = CC_C, | |
2883 | + [INDEX_op_jb_subl] = CC_C, | |
2884 | + | |
2885 | + [INDEX_op_jz_subb] = CC_Z, | |
2886 | + [INDEX_op_jz_subw] = CC_Z, | |
2887 | + [INDEX_op_jz_subl] = CC_Z, | |
2888 | + | |
2889 | + [INDEX_op_jbe_subb] = CC_Z | CC_C, | |
2890 | + [INDEX_op_jbe_subw] = CC_Z | CC_C, | |
2891 | + [INDEX_op_jbe_subl] = CC_Z | CC_C, | |
2892 | + | |
2893 | + [INDEX_op_js_subb] = CC_S, | |
2894 | + [INDEX_op_js_subw] = CC_S, | |
2895 | + [INDEX_op_js_subl] = CC_S, | |
2896 | + | |
2897 | + [INDEX_op_jl_subb] = CC_O | CC_S, | |
2898 | + [INDEX_op_jl_subw] = CC_O | CC_S, | |
2899 | + [INDEX_op_jl_subl] = CC_O | CC_S, | |
2900 | + | |
2901 | + [INDEX_op_jle_subb] = CC_O | CC_S | CC_Z, | |
2902 | + [INDEX_op_jle_subw] = CC_O | CC_S | CC_Z, | |
2903 | + [INDEX_op_jle_subl] = CC_O | CC_S | CC_Z, | |
2904 | + | |
2905 | + [INDEX_op_loopnzw] = CC_Z, | |
2906 | + [INDEX_op_loopnzl] = CC_Z, | |
2907 | + [INDEX_op_loopzw] = CC_Z, | |
2908 | + [INDEX_op_loopzl] = CC_Z, | |
2909 | + | |
2910 | + [INDEX_op_seto_T0_cc] = CC_O, | |
2911 | + [INDEX_op_setb_T0_cc] = CC_C, | |
2912 | + [INDEX_op_setz_T0_cc] = CC_Z, | |
2913 | + [INDEX_op_setbe_T0_cc] = CC_Z | CC_C, | |
2914 | + [INDEX_op_sets_T0_cc] = CC_S, | |
2915 | + [INDEX_op_setp_T0_cc] = CC_P, | |
2916 | + [INDEX_op_setl_T0_cc] = CC_O | CC_S, | |
2917 | + [INDEX_op_setle_T0_cc] = CC_O | CC_S | CC_Z, | |
2918 | + | |
2919 | + [INDEX_op_setb_T0_subb] = CC_C, | |
2920 | + [INDEX_op_setb_T0_subw] = CC_C, | |
2921 | + [INDEX_op_setb_T0_subl] = CC_C, | |
2922 | + | |
2923 | + [INDEX_op_setz_T0_subb] = CC_Z, | |
2924 | + [INDEX_op_setz_T0_subw] = CC_Z, | |
2925 | + [INDEX_op_setz_T0_subl] = CC_Z, | |
2926 | + | |
2927 | + [INDEX_op_setbe_T0_subb] = CC_Z | CC_C, | |
2928 | + [INDEX_op_setbe_T0_subw] = CC_Z | CC_C, | |
2929 | + [INDEX_op_setbe_T0_subl] = CC_Z | CC_C, | |
2930 | + | |
2931 | + [INDEX_op_sets_T0_subb] = CC_S, | |
2932 | + [INDEX_op_sets_T0_subw] = CC_S, | |
2933 | + [INDEX_op_sets_T0_subl] = CC_S, | |
2934 | + | |
2935 | + [INDEX_op_setl_T0_subb] = CC_O | CC_S, | |
2936 | + [INDEX_op_setl_T0_subw] = CC_O | CC_S, | |
2937 | + [INDEX_op_setl_T0_subl] = CC_O | CC_S, | |
2938 | + | |
2939 | + [INDEX_op_setle_T0_subb] = CC_O | CC_S | CC_Z, | |
2940 | + [INDEX_op_setle_T0_subw] = CC_O | CC_S | CC_Z, | |
2941 | + [INDEX_op_setle_T0_subl] = CC_O | CC_S | CC_Z, | |
2942 | + | |
2943 | + [INDEX_op_movl_T0_eflags] = CC_OSZAPC, | |
2944 | + [INDEX_op_cmc] = CC_C, | |
2945 | + [INDEX_op_salc] = CC_C, | |
2946 | + | |
2947 | + [INDEX_op_rclb_T0_T1_cc] = CC_C, | |
2948 | + [INDEX_op_rclw_T0_T1_cc] = CC_C, | |
2949 | + [INDEX_op_rcll_T0_T1_cc] = CC_C, | |
2950 | + [INDEX_op_rcrb_T0_T1_cc] = CC_C, | |
2951 | + [INDEX_op_rcrw_T0_T1_cc] = CC_C, | |
2952 | + [INDEX_op_rcrl_T0_T1_cc] = CC_C, | |
2953 | +}; | |
2954 | + | |
2955 | +/* flags written by an operation */ | |
2956 | +static uint16_t opc_write_flags[NB_OPS] = { | |
2957 | + [INDEX_op_addl_T0_T1_cc] = CC_OSZAPC, | |
2958 | + [INDEX_op_orl_T0_T1_cc] = CC_OSZAPC, | |
2959 | + [INDEX_op_adcb_T0_T1_cc] = CC_OSZAPC, | |
2960 | + [INDEX_op_adcw_T0_T1_cc] = CC_OSZAPC, | |
2961 | + [INDEX_op_adcl_T0_T1_cc] = CC_OSZAPC, | |
2962 | + [INDEX_op_sbbb_T0_T1_cc] = CC_OSZAPC, | |
2963 | + [INDEX_op_sbbw_T0_T1_cc] = CC_OSZAPC, | |
2964 | + [INDEX_op_sbbl_T0_T1_cc] = CC_OSZAPC, | |
2965 | + [INDEX_op_andl_T0_T1_cc] = CC_OSZAPC, | |
2966 | + [INDEX_op_subl_T0_T1_cc] = CC_OSZAPC, | |
2967 | + [INDEX_op_xorl_T0_T1_cc] = CC_OSZAPC, | |
2968 | + [INDEX_op_cmpl_T0_T1_cc] = CC_OSZAPC, | |
2969 | + [INDEX_op_negl_T0_cc] = CC_OSZAPC, | |
2970 | + [INDEX_op_incl_T0_cc] = CC_OSZAP, | |
2971 | + [INDEX_op_decl_T0_cc] = CC_OSZAP, | |
2972 | + [INDEX_op_testl_T0_T1_cc] = CC_OSZAPC, | |
2973 | + | |
2974 | + [INDEX_op_mulb_AL_T0] = CC_OSZAPC, | |
2975 | + [INDEX_op_imulb_AL_T0] = CC_OSZAPC, | |
2976 | + [INDEX_op_mulw_AX_T0] = CC_OSZAPC, | |
2977 | + [INDEX_op_imulw_AX_T0] = CC_OSZAPC, | |
2978 | + [INDEX_op_mull_EAX_T0] = CC_OSZAPC, | |
2979 | + [INDEX_op_imull_EAX_T0] = CC_OSZAPC, | |
2980 | + [INDEX_op_imulw_T0_T1] = CC_OSZAPC, | |
2981 | + [INDEX_op_imull_T0_T1] = CC_OSZAPC, | |
2982 | + | |
2983 | + /* bcd */ | |
2984 | + [INDEX_op_aam] = CC_OSZAPC, | |
2985 | + [INDEX_op_aad] = CC_OSZAPC, | |
2986 | + [INDEX_op_aas] = CC_OSZAPC, | |
2987 | + [INDEX_op_aaa] = CC_OSZAPC, | |
2988 | + [INDEX_op_das] = CC_OSZAPC, | |
2989 | + [INDEX_op_daa] = CC_OSZAPC, | |
2990 | + | |
2991 | + [INDEX_op_movb_eflags_T0] = CC_S | CC_Z | CC_A | CC_P | CC_C, | |
2992 | + [INDEX_op_movl_eflags_T0] = CC_OSZAPC, | |
2993 | + [INDEX_op_clc] = CC_C, | |
2994 | + [INDEX_op_stc] = CC_C, | |
2995 | + [INDEX_op_cmc] = CC_C, | |
2996 | + | |
2997 | + [INDEX_op_rolb_T0_T1_cc] = CC_O | CC_C, | |
2998 | + [INDEX_op_rolw_T0_T1_cc] = CC_O | CC_C, | |
2999 | + [INDEX_op_roll_T0_T1_cc] = CC_O | CC_C, | |
3000 | + [INDEX_op_rorb_T0_T1_cc] = CC_O | CC_C, | |
3001 | + [INDEX_op_rorw_T0_T1_cc] = CC_O | CC_C, | |
3002 | + [INDEX_op_rorl_T0_T1_cc] = CC_O | CC_C, | |
3003 | + | |
3004 | + [INDEX_op_rclb_T0_T1_cc] = CC_O | CC_C, | |
3005 | + [INDEX_op_rclw_T0_T1_cc] = CC_O | CC_C, | |
3006 | + [INDEX_op_rcll_T0_T1_cc] = CC_O | CC_C, | |
3007 | + [INDEX_op_rcrb_T0_T1_cc] = CC_O | CC_C, | |
3008 | + [INDEX_op_rcrw_T0_T1_cc] = CC_O | CC_C, | |
3009 | + [INDEX_op_rcrl_T0_T1_cc] = CC_O | CC_C, | |
3010 | + | |
3011 | + [INDEX_op_shlb_T0_T1_cc] = CC_OSZAPC, | |
3012 | + [INDEX_op_shlw_T0_T1_cc] = CC_OSZAPC, | |
3013 | + [INDEX_op_shll_T0_T1_cc] = CC_OSZAPC, | |
3014 | + | |
3015 | + [INDEX_op_shrb_T0_T1_cc] = CC_OSZAPC, | |
3016 | + [INDEX_op_shrw_T0_T1_cc] = CC_OSZAPC, | |
3017 | + [INDEX_op_shrl_T0_T1_cc] = CC_OSZAPC, | |
3018 | + | |
3019 | + [INDEX_op_sarb_T0_T1_cc] = CC_OSZAPC, | |
3020 | + [INDEX_op_sarw_T0_T1_cc] = CC_OSZAPC, | |
3021 | + [INDEX_op_sarl_T0_T1_cc] = CC_OSZAPC, | |
3022 | + | |
3023 | + [INDEX_op_shldw_T0_T1_ECX_cc] = CC_OSZAPC, | |
3024 | + [INDEX_op_shldl_T0_T1_ECX_cc] = CC_OSZAPC, | |
3025 | + [INDEX_op_shldw_T0_T1_im_cc] = CC_OSZAPC, | |
3026 | + [INDEX_op_shldl_T0_T1_im_cc] = CC_OSZAPC, | |
3027 | + | |
3028 | + [INDEX_op_shrdw_T0_T1_ECX_cc] = CC_OSZAPC, | |
3029 | + [INDEX_op_shrdl_T0_T1_ECX_cc] = CC_OSZAPC, | |
3030 | + [INDEX_op_shrdw_T0_T1_im_cc] = CC_OSZAPC, | |
3031 | + [INDEX_op_shrdl_T0_T1_im_cc] = CC_OSZAPC, | |
3032 | + | |
3033 | + [INDEX_op_btw_T0_T1_cc] = CC_OSZAPC, | |
3034 | + [INDEX_op_btl_T0_T1_cc] = CC_OSZAPC, | |
3035 | + [INDEX_op_btsw_T0_T1_cc] = CC_OSZAPC, | |
3036 | + [INDEX_op_btsl_T0_T1_cc] = CC_OSZAPC, | |
3037 | + [INDEX_op_btrw_T0_T1_cc] = CC_OSZAPC, | |
3038 | + [INDEX_op_btrl_T0_T1_cc] = CC_OSZAPC, | |
3039 | + [INDEX_op_btcw_T0_T1_cc] = CC_OSZAPC, | |
3040 | + [INDEX_op_btcl_T0_T1_cc] = CC_OSZAPC, | |
3041 | + | |
3042 | + [INDEX_op_bsfw_T0_cc] = CC_OSZAPC, | |
3043 | + [INDEX_op_bsfl_T0_cc] = CC_OSZAPC, | |
3044 | + [INDEX_op_bsrw_T0_cc] = CC_OSZAPC, | |
3045 | + [INDEX_op_bsrl_T0_cc] = CC_OSZAPC, | |
3046 | + | |
3047 | + [INDEX_op_scasb] = CC_OSZAPC, | |
3048 | + [INDEX_op_scasw] = CC_OSZAPC, | |
3049 | + [INDEX_op_scasl] = CC_OSZAPC, | |
3050 | + [INDEX_op_repz_scasb] = CC_OSZAPC, | |
3051 | + [INDEX_op_repz_scasw] = CC_OSZAPC, | |
3052 | + [INDEX_op_repz_scasl] = CC_OSZAPC, | |
3053 | + [INDEX_op_repnz_scasb] = CC_OSZAPC, | |
3054 | + [INDEX_op_repnz_scasw] = CC_OSZAPC, | |
3055 | + [INDEX_op_repnz_scasl] = CC_OSZAPC, | |
3056 | + | |
3057 | + [INDEX_op_cmpsb] = CC_OSZAPC, | |
3058 | + [INDEX_op_cmpsw] = CC_OSZAPC, | |
3059 | + [INDEX_op_cmpsl] = CC_OSZAPC, | |
3060 | + [INDEX_op_repz_cmpsb] = CC_OSZAPC, | |
3061 | + [INDEX_op_repz_cmpsw] = CC_OSZAPC, | |
3062 | + [INDEX_op_repz_cmpsl] = CC_OSZAPC, | |
3063 | + [INDEX_op_repnz_cmpsb] = CC_OSZAPC, | |
3064 | + [INDEX_op_repnz_cmpsw] = CC_OSZAPC, | |
3065 | + [INDEX_op_repnz_cmpsl] = CC_OSZAPC, | |
3066 | + | |
3067 | + [INDEX_op_cmpxchgw_T0_T1_EAX_cc] = CC_OSZAPC, | |
3068 | + [INDEX_op_cmpxchgl_T0_T1_EAX_cc] = CC_OSZAPC, | |
3069 | +}; | |
3070 | + | |
3071 | +/* simpler form of an operation if no flags need to be generated */ | |
3072 | +static uint16_t opc_simpler[NB_OPS] = { | |
3073 | + [INDEX_op_addl_T0_T1_cc] = INDEX_op_addl_T0_T1, | |
3074 | + [INDEX_op_orl_T0_T1_cc] = INDEX_op_orl_T0_T1, | |
3075 | + [INDEX_op_andl_T0_T1_cc] = INDEX_op_andl_T0_T1, | |
3076 | + [INDEX_op_subl_T0_T1_cc] = INDEX_op_subl_T0_T1, | |
3077 | + [INDEX_op_xorl_T0_T1_cc] = INDEX_op_xorl_T0_T1, | |
3078 | + [INDEX_op_negl_T0_cc] = INDEX_op_negl_T0, | |
3079 | + [INDEX_op_incl_T0_cc] = INDEX_op_incl_T0, | |
3080 | + [INDEX_op_decl_T0_cc] = INDEX_op_decl_T0, | |
3081 | + | |
3082 | + [INDEX_op_rolb_T0_T1_cc] = INDEX_op_rolb_T0_T1, | |
3083 | + [INDEX_op_rolw_T0_T1_cc] = INDEX_op_rolw_T0_T1, | |
3084 | + [INDEX_op_roll_T0_T1_cc] = INDEX_op_roll_T0_T1, | |
3085 | + | |
3086 | + [INDEX_op_rorb_T0_T1_cc] = INDEX_op_rorb_T0_T1, | |
3087 | + [INDEX_op_rorw_T0_T1_cc] = INDEX_op_rorw_T0_T1, | |
3088 | + [INDEX_op_rorl_T0_T1_cc] = INDEX_op_rorl_T0_T1, | |
3089 | + | |
3090 | + [INDEX_op_shlb_T0_T1_cc] = INDEX_op_shlb_T0_T1, | |
3091 | + [INDEX_op_shlw_T0_T1_cc] = INDEX_op_shlw_T0_T1, | |
3092 | + [INDEX_op_shll_T0_T1_cc] = INDEX_op_shll_T0_T1, | |
3093 | + | |
3094 | + [INDEX_op_shrb_T0_T1_cc] = INDEX_op_shrb_T0_T1, | |
3095 | + [INDEX_op_shrw_T0_T1_cc] = INDEX_op_shrw_T0_T1, | |
3096 | + [INDEX_op_shrl_T0_T1_cc] = INDEX_op_shrl_T0_T1, | |
3097 | + | |
3098 | + [INDEX_op_sarb_T0_T1_cc] = INDEX_op_sarb_T0_T1, | |
3099 | + [INDEX_op_sarw_T0_T1_cc] = INDEX_op_sarw_T0_T1, | |
3100 | + [INDEX_op_sarl_T0_T1_cc] = INDEX_op_sarl_T0_T1, | |
3101 | +}; | |
3102 | + | |
3103 | +static void optimize_flags_init(void) | |
3104 | +{ | |
3105 | + int i; | |
3106 | + /* put default values in arrays */ | |
3107 | + for(i = 0; i < NB_OPS; i++) { | |
3108 | + if (opc_simpler[i] == 0) | |
3109 | + opc_simpler[i] = i; | |
3110 | + } | |
3111 | +} | |
3112 | + | |
3113 | +/* CPU flags computation optimization: we move backward thru the | |
3114 | + generated code to see which flags are needed. The operation is | |
3115 | + modified if suitable */ | |
3116 | +static void optimize_flags(uint16_t *opc_buf, int opc_buf_len) | |
3117 | +{ | |
3118 | + uint16_t *opc_ptr; | |
3119 | + int live_flags, write_flags, op; | |
3120 | + | |
3121 | + opc_ptr = opc_buf + opc_buf_len; | |
3122 | + /* live_flags contains the flags needed by the next instructions | |
3123 | + in the code. At the end of the bloc, we consider that all the | |
3124 | + flags are live. */ | |
3125 | + live_flags = CC_OSZAPC; | |
3126 | + while (opc_ptr > opc_buf) { | |
3127 | + op = *--opc_ptr; | |
3128 | + /* if none of the flags written by the instruction is used, | |
3129 | + then we can try to find a simpler instruction */ | |
3130 | + write_flags = opc_write_flags[op]; | |
3131 | + if ((live_flags & write_flags) == 0) { | |
3132 | + *opc_ptr = opc_simpler[op]; | |
3133 | + } | |
3134 | + /* compute the live flags before the instruction */ | |
3135 | + live_flags &= ~write_flags; | |
3136 | + live_flags |= opc_read_flags[op]; | |
3137 | + } | |
3138 | +} | |
3139 | + | |
3140 | + | |
3141 | +#ifdef DEBUG_DISAS | |
3142 | +static const char *op_str[] = { | |
3143 | +#define DEF(s) #s, | |
3144 | +#include "opc-i386.h" | |
3145 | +#undef DEF | |
3146 | +}; | |
3147 | + | |
3148 | +static void dump_ops(const uint16_t *opc_buf) | |
3149 | +{ | |
3150 | + const uint16_t *opc_ptr; | |
3151 | + int c; | |
3152 | + opc_ptr = opc_buf; | |
3153 | + for(;;) { | |
3154 | + c = *opc_ptr++; | |
3155 | + fprintf(logfile, "0x%04x: %s\n", opc_ptr - opc_buf - 1, op_str[c]); | |
3156 | + if (c == INDEX_op_end) | |
3157 | + break; | |
3158 | + } | |
3159 | +} | |
3160 | + | |
3161 | +#endif | |
3162 | + | |
3163 | +/* XXX: make this buffer thread safe */ | |
3164 | +/* XXX: make safe guess about sizes */ | |
3165 | +#define MAX_OP_PER_INSTR 32 | |
3166 | +#define OPC_BUF_SIZE 512 | |
3167 | +#define OPC_MAX_SIZE (OPC_BUF_SIZE - MAX_OP_PER_INSTR) | |
3168 | + | |
3169 | +#define OPPARAM_BUF_SIZE (OPC_BUF_SIZE * 3) | |
3170 | + | |
3171 | +static uint16_t gen_opc_buf[OPC_BUF_SIZE]; | |
3172 | +static uint32_t gen_opparam_buf[OPPARAM_BUF_SIZE]; | |
3173 | + | |
2845 | 3174 | /* return the next pc */ |
2846 | 3175 | int cpu_x86_gen_code(uint8_t *gen_code_buf, int max_code_size, |
2847 | 3176 | int *gen_code_size_ptr, uint8_t *pc_start, |
2848 | 3177 | int flags) |
2849 | 3178 | { |
2850 | 3179 | DisasContext dc1, *dc = &dc1; |
2851 | - uint8_t *gen_code_end, *pc_ptr; | |
3180 | + uint8_t *pc_ptr; | |
3181 | + uint16_t *gen_opc_end; | |
2852 | 3182 | long ret; |
2853 | 3183 | #ifdef DEBUG_DISAS |
2854 | 3184 | struct disassemble_info disasm_info; |
2855 | 3185 | #endif |
3186 | + | |
3187 | + /* generate intermediate code */ | |
3188 | + | |
2856 | 3189 | dc->code32 = (flags >> GEN_FLAG_CODE32_SHIFT) & 1; |
2857 | 3190 | dc->addseg = (flags >> GEN_FLAG_ADDSEG_SHIFT) & 1; |
2858 | 3191 | dc->f_st = (flags >> GEN_FLAG_ST_SHIFT) & 7; |
2859 | 3192 | dc->cc_op = CC_OP_DYNAMIC; |
2860 | - gen_code_ptr = gen_code_buf; | |
2861 | - gen_code_end = gen_code_buf + max_code_size - 4096; | |
2862 | - gen_start(); | |
3193 | + | |
3194 | + gen_opc_ptr = gen_opc_buf; | |
3195 | + gen_opc_end = gen_opc_buf + OPC_MAX_SIZE; | |
3196 | + gen_opparam_ptr = gen_opparam_buf; | |
2863 | 3197 | |
2864 | 3198 | dc->is_jmp = 0; |
2865 | 3199 | pc_ptr = pc_start; |
... | ... | @@ -2871,7 +3205,7 @@ int cpu_x86_gen_code(uint8_t *gen_code_buf, int max_code_size, |
2871 | 3205 | abort(); |
2872 | 3206 | } |
2873 | 3207 | pc_ptr = (void *)ret; |
2874 | - } while (!dc->is_jmp && gen_code_ptr < gen_code_end); | |
3208 | + } while (!dc->is_jmp && gen_opc_ptr < gen_opc_end); | |
2875 | 3209 | /* we must store the eflags state if it is not already done */ |
2876 | 3210 | if (dc->cc_op != CC_OP_DYNAMIC) |
2877 | 3211 | gen_op_set_cc_op(dc->cc_op); |
... | ... | @@ -2879,9 +3213,9 @@ int cpu_x86_gen_code(uint8_t *gen_code_buf, int max_code_size, |
2879 | 3213 | /* we add an additionnal jmp to update the simulated PC */ |
2880 | 3214 | gen_op_jmp_im(ret); |
2881 | 3215 | } |
2882 | - gen_end(); | |
2883 | - *gen_code_size_ptr = gen_code_ptr - gen_code_buf; | |
3216 | + *gen_opc_ptr = INDEX_op_end; | |
2884 | 3217 | |
3218 | + /* optimize flag computations */ | |
2885 | 3219 | #ifdef DEBUG_DISAS |
2886 | 3220 | if (loglevel) { |
2887 | 3221 | uint8_t *pc; |
... | ... | @@ -2898,6 +3232,7 @@ int cpu_x86_gen_code(uint8_t *gen_code_buf, int max_code_size, |
2898 | 3232 | #else |
2899 | 3233 | disasm_info.endian = BFD_ENDIAN_LITTLE; |
2900 | 3234 | #endif |
3235 | + fprintf(logfile, "----------------\n"); | |
2901 | 3236 | fprintf(logfile, "IN:\n"); |
2902 | 3237 | disasm_info.buffer = pc_start; |
2903 | 3238 | disasm_info.buffer_vma = (unsigned long)pc_start; |
... | ... | @@ -2911,12 +3246,37 @@ int cpu_x86_gen_code(uint8_t *gen_code_buf, int max_code_size, |
2911 | 3246 | } |
2912 | 3247 | fprintf(logfile, "\n"); |
2913 | 3248 | |
3249 | + fprintf(logfile, "OP:\n"); | |
3250 | + dump_ops(gen_opc_buf); | |
3251 | + fprintf(logfile, "\n"); | |
3252 | + } | |
3253 | +#endif | |
3254 | + | |
3255 | + /* optimize flag computations */ | |
3256 | + optimize_flags(gen_opc_buf, gen_opc_ptr - gen_opc_buf); | |
3257 | + | |
3258 | +#ifdef DEBUG_DISAS | |
3259 | + if (loglevel) { | |
3260 | + fprintf(logfile, "AFTER FLAGS OPT:\n"); | |
3261 | + dump_ops(gen_opc_buf); | |
3262 | + fprintf(logfile, "\n"); | |
3263 | + } | |
3264 | +#endif | |
3265 | + | |
3266 | + /* generate machine code */ | |
3267 | + *gen_code_size_ptr = dyngen_code(gen_code_buf, gen_opc_buf, gen_opparam_buf); | |
3268 | + | |
3269 | +#ifdef DEBUG_DISAS | |
3270 | + if (loglevel) { | |
3271 | + uint8_t *pc; | |
3272 | + int count; | |
3273 | + | |
2914 | 3274 | pc = gen_code_buf; |
2915 | 3275 | disasm_info.buffer = pc; |
2916 | 3276 | disasm_info.buffer_vma = (unsigned long)pc; |
2917 | 3277 | disasm_info.buffer_length = *gen_code_size_ptr; |
2918 | 3278 | fprintf(logfile, "OUT: [size=%d]\n", *gen_code_size_ptr); |
2919 | - while (pc < gen_code_ptr) { | |
3279 | + while (pc < gen_code_buf + *gen_code_size_ptr) { | |
2920 | 3280 | fprintf(logfile, "0x%08lx: ", (long)pc); |
2921 | 3281 | count = print_insn_i386((unsigned long)pc, &disasm_info); |
2922 | 3282 | fprintf(logfile, "\n"); |
... | ... | @@ -2932,6 +3292,7 @@ CPUX86State *cpu_x86_init(void) |
2932 | 3292 | { |
2933 | 3293 | CPUX86State *env; |
2934 | 3294 | int i; |
3295 | + static int inited; | |
2935 | 3296 | |
2936 | 3297 | cpu_x86_tblocks_init(); |
2937 | 3298 | |
... | ... | @@ -2946,6 +3307,12 @@ CPUX86State *cpu_x86_init(void) |
2946 | 3307 | /* flags setup */ |
2947 | 3308 | env->cc_op = CC_OP_EFLAGS; |
2948 | 3309 | env->df = 1; |
3310 | + | |
3311 | + /* init various static tables */ | |
3312 | + if (!inited) { | |
3313 | + inited = 1; | |
3314 | + optimize_flags_init(); | |
3315 | + } | |
2949 | 3316 | return env; |
2950 | 3317 | } |
2951 | 3318 | ... | ... |