]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - usr.bin/doscmd/i386-pinsn.c
This commit was generated by cvs2svn to compensate for changes in r76238,
[FreeBSD/FreeBSD.git] / usr.bin / doscmd / i386-pinsn.c
1 #ifdef DISASSEMBLER
2
3 /* Print i386 instructions for GDB, the GNU debugger.
4    Copyright (C) 1988, 1989 Free Software Foundation, Inc.
5
6 This file is part of GDB.
7
8 GDB is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 1, or (at your option)
11 any later version.
12
13 GDB is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 GNU General Public License for more details.
17
18 You should have received a copy of the GNU General Public License
19 along with GDB; see the file COPYING.  If not, write to
20 the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
21
22 /*
23  * 80386 instruction printer by Pace Willisson (pace@prep.ai.mit.edu)
24  * July 1988
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 <stdio.h>
37 #include <ctype.h>
38
39 #define Eb OP_E, b_mode
40 #define indirEb OP_indirE, b_mode
41 #define Gb OP_G, b_mode
42 #define Ev OP_E, v_mode
43 #define indirEv OP_indirE, v_mode
44 #define Ew OP_E, w_mode
45 #define Ma OP_E, v_mode
46 #define M OP_E, 0
47 #define Mp OP_E, 0              /* ? */
48 #define Gv OP_G, v_mode
49 #define Gw OP_G, w_mode
50 #define Rw OP_rm, w_mode
51 #define Rd OP_rm, d_mode
52 #define Ib OP_I, b_mode
53 #define sIb OP_sI, b_mode       /* sign extened byte */
54 #define Iv OP_I, v_mode
55 #define Iw OP_I, w_mode
56 #define Jb OP_J, b_mode
57 #define Jv OP_J, v_mode
58 #define ONE OP_ONE, 0
59 #define Cd OP_C, d_mode
60 #define Dd OP_D, d_mode
61 #define Td OP_T, d_mode
62
63 #define eAX OP_REG, eAX_reg
64 #define eBX OP_REG, eBX_reg
65 #define eCX OP_REG, eCX_reg
66 #define eDX OP_REG, eDX_reg
67 #define eSP OP_REG, eSP_reg
68 #define eBP OP_REG, eBP_reg
69 #define eSI OP_REG, eSI_reg
70 #define eDI OP_REG, eDI_reg
71 #define AL OP_REG, al_reg
72 #define CL OP_REG, cl_reg
73 #define DL OP_REG, dl_reg
74 #define BL OP_REG, bl_reg
75 #define AH OP_REG, ah_reg
76 #define CH OP_REG, ch_reg
77 #define DH OP_REG, dh_reg
78 #define BH OP_REG, bh_reg
79 #define AX OP_REG, ax_reg
80 #define DX OP_REG, dx_reg
81 #define indirDX OP_REG, indir_dx_reg
82
83 #define Sw OP_SEG, w_mode
84 #define Ap OP_DIR, lptr
85 #define Av OP_DIR, v_mode
86 #define Ob OP_OFF, b_mode
87 #define Ov OP_OFF, v_mode
88 #define Xb OP_DSSI, b_mode
89 #define Xv OP_DSSI, v_mode
90 #define Yb OP_ESDI, b_mode
91 #define Yv OP_ESDI, v_mode
92
93 #define es OP_REG, es_reg
94 #define ss OP_REG, ss_reg
95 #define cs OP_REG, cs_reg
96 #define ds OP_REG, ds_reg
97 #define fs OP_REG, fs_reg
98 #define gs OP_REG, gs_reg
99
100 int OP_E(), OP_indirE(), OP_G(), OP_I(), OP_sI(), OP_REG();
101 int OP_J(), OP_SEG();
102 int OP_DIR(), OP_OFF(), OP_DSSI(), OP_ESDI(), OP_ONE(), OP_C();
103 int OP_D(), OP_T(), OP_rm();
104
105
106 #define b_mode 1
107 #define v_mode 2
108 #define w_mode 3
109 #define d_mode 4
110
111 #define es_reg 100
112 #define cs_reg 101
113 #define ss_reg 102
114 #define ds_reg 103
115 #define fs_reg 104
116 #define gs_reg 105
117 #define eAX_reg 107
118 #define eCX_reg 108
119 #define eDX_reg 109
120 #define eBX_reg 110
121 #define eSP_reg 111
122 #define eBP_reg 112
123 #define eSI_reg 113
124 #define eDI_reg 114
125
126 #define lptr 115
127
128 #define al_reg 116
129 #define cl_reg 117
130 #define dl_reg 118
131 #define bl_reg 119
132 #define ah_reg 120
133 #define ch_reg 121
134 #define dh_reg 122
135 #define bh_reg 123
136
137 #define ax_reg 124
138 #define cx_reg 125
139 #define dx_reg 126
140 #define bx_reg 127
141 #define sp_reg 128
142 #define bp_reg 129
143 #define si_reg 130
144 #define di_reg 131
145
146 #define indir_dx_reg 150
147
148 #define GRP1b NULL, NULL, 0
149 #define GRP1S NULL, NULL, 1
150 #define GRP1Ss NULL, NULL, 2
151 #define GRP2b NULL, NULL, 3
152 #define GRP2S NULL, NULL, 4
153 #define GRP2b_one NULL, NULL, 5
154 #define GRP2S_one NULL, NULL, 6
155 #define GRP2b_cl NULL, NULL, 7
156 #define GRP2S_cl NULL, NULL, 8
157 #define GRP3b NULL, NULL, 9
158 #define GRP3S NULL, NULL, 10
159 #define GRP4  NULL, NULL, 11
160 #define GRP5  NULL, NULL, 12
161 #define GRP6  NULL, NULL, 13
162 #define GRP7 NULL, NULL, 14
163 #define GRP8 NULL, NULL, 15
164
165 #define FLOATCODE 50
166 #define FLOAT NULL, NULL, FLOATCODE
167
168 struct dis386 {
169   char *name;
170   int (*op1)();
171   int bytemode1;
172   int (*op2)();
173   int bytemode2;
174   int (*op3)();
175   int bytemode3;
176 };
177
178 struct dis386 dis386[] = {
179   /* 00 */
180   { "addb",     Eb, Gb },
181   { "addS",     Ev, Gv },
182   { "addb",     Gb, Eb },
183   { "addS",     Gv, Ev },
184   { "addb",     AL, Ib },
185   { "addS",     eAX, Iv },
186   { "pushl",    es },
187   { "popl",     es },
188   /* 08 */
189   { "orb",      Eb, Gb },
190   { "orS",      Ev, Gv },
191   { "orb",      Gb, Eb },
192   { "orS",      Gv, Ev },
193   { "orb",      AL, Ib },
194   { "orS",      eAX, Iv },
195   { "pushl",    cs },
196   { "(bad)" },  /* 0x0f extended opcode escape */
197   /* 10 */
198   { "adcb",     Eb, Gb },
199   { "adcS",     Ev, Gv },
200   { "adcb",     Gb, Eb },
201   { "adcS",     Gv, Ev },
202   { "adcb",     AL, Ib },
203   { "adcS",     eAX, Iv },
204   { "pushl",    ss },
205   { "popl",     ss },
206   /* 18 */
207   { "sbbb",     Eb, Gb },
208   { "sbbS",     Ev, Gv },
209   { "sbbb",     Gb, Eb },
210   { "sbbS",     Gv, Ev },
211   { "sbbb",     AL, Ib },
212   { "sbbS",     eAX, Iv },
213   { "pushl",    ds },
214   { "popl",     ds },
215   /* 20 */
216   { "andb",     Eb, Gb },
217   { "andS",     Ev, Gv },
218   { "andb",     Gb, Eb },
219   { "andS",     Gv, Ev },
220   { "andb",     AL, Ib },
221   { "andS",     eAX, Iv },
222   { "(bad)" },                  /* SEG ES prefix */
223   { "daa" },
224   /* 28 */
225   { "subb",     Eb, Gb },
226   { "subS",     Ev, Gv },
227   { "subb",     Gb, Eb },
228   { "subS",     Gv, Ev },
229   { "subb",     AL, Ib },
230   { "subS",     eAX, Iv },
231   { "(bad)" },                  /* SEG CS prefix */
232   { "das" },
233   /* 30 */
234   { "xorb",     Eb, Gb },
235   { "xorS",     Ev, Gv },
236   { "xorb",     Gb, Eb },
237   { "xorS",     Gv, Ev },
238   { "xorb",     AL, Ib },
239   { "xorS",     eAX, Iv },
240   { "(bad)" },                  /* SEG SS prefix */
241   { "aaa" },
242   /* 38 */
243   { "cmpb",     Eb, Gb },
244   { "cmpS",     Ev, Gv },
245   { "cmpb",     Gb, Eb },
246   { "cmpS",     Gv, Ev },
247   { "cmpb",     AL, Ib },
248   { "cmpS",     eAX, Iv },
249   { "(bad)" },                  /* SEG DS prefix */
250   { "aas" },
251   /* 40 */
252   { "incS",     eAX },
253   { "incS",     eCX },
254   { "incS",     eDX },
255   { "incS",     eBX },
256   { "incS",     eSP },
257   { "incS",     eBP },
258   { "incS",     eSI },
259   { "incS",     eDI },
260   /* 48 */
261   { "decS",     eAX },
262   { "decS",     eCX },
263   { "decS",     eDX },
264   { "decS",     eBX },
265   { "decS",     eSP },
266   { "decS",     eBP },
267   { "decS",     eSI },
268   { "decS",     eDI },
269   /* 50 */
270   { "pushS",    eAX },
271   { "pushS",    eCX },
272   { "pushS",    eDX },
273   { "pushS",    eBX },
274   { "pushS",    eSP },
275   { "pushS",    eBP },
276   { "pushS",    eSI },
277   { "pushS",    eDI },
278   /* 58 */
279   { "popS",     eAX },
280   { "popS",     eCX },
281   { "popS",     eDX },
282   { "popS",     eBX },
283   { "popS",     eSP },
284   { "popS",     eBP },
285   { "popS",     eSI },
286   { "popS",     eDI },
287   /* 60 */
288   { "pusha" },
289   { "popa" },
290   { "boundS",   Gv, Ma },
291   { "arpl",     Ew, Gw },
292   { "(bad)" },                  /* seg fs */
293   { "(bad)" },                  /* seg gs */
294   { "(bad)" },                  /* op size prefix */
295   { "(bad)" },                  /* adr size prefix */
296   /* 68 */
297   { "pushS",    Iv },           /* 386 book wrong */
298   { "imulS",    Gv, Ev, Iv },
299   { "pushl",    sIb },          /* push of byte really pushes 4 bytes */
300   { "imulS",    Gv, Ev, Ib },
301   { "insb",     Yb, indirDX },
302   { "insS",     Yv, indirDX },
303   { "outsb",    indirDX, Xb },
304   { "outsS",    indirDX, Xv },
305   /* 70 */
306   { "jo",               Jb },
307   { "jno",      Jb },
308   { "jb",               Jb },
309   { "jae",      Jb },
310   { "je",               Jb },
311   { "jne",      Jb },
312   { "jbe",      Jb },
313   { "ja",               Jb },
314   /* 78 */
315   { "js",               Jb },
316   { "jns",      Jb },
317   { "jp",               Jb },
318   { "jnp",      Jb },
319   { "jl",               Jb },
320   { "jnl",      Jb },
321   { "jle",      Jb },
322   { "jg",               Jb },
323   /* 80 */
324   { GRP1b },
325   { GRP1S },
326   { "(bad)" },
327   { GRP1Ss },
328   { "testb",    Eb, Gb },
329   { "testS",    Ev, Gv },
330   { "xchgb",    Eb, Gb },
331   { "xchgS",    Ev, Gv },
332   /* 88 */
333   { "movb",     Eb, Gb },
334   { "movS",     Ev, Gv },
335   { "movb",     Gb, Eb },
336   { "movS",     Gv, Ev },
337   { "movw",     Ew, Sw },
338   { "leaS",     Gv, M },
339   { "movw",     Sw, Ew },
340   { "popS",     Ev },
341   /* 90 */
342   { "nop" },
343   { "xchgS",    eCX, eAX },
344   { "xchgS",    eDX, eAX },
345   { "xchgS",    eBX, eAX },
346   { "xchgS",    eSP, eAX },
347   { "xchgS",    eBP, eAX },
348   { "xchgS",    eSI, eAX },
349   { "xchgS",    eDI, eAX },
350   /* 98 */
351   { "cwtl" },
352   { "cltd" },
353   { "lcall",    Ap },
354   { "(bad)" },          /* fwait */
355   { "pushf" },
356   { "popf" },
357   { "sahf" },
358   { "lahf" },
359   /* a0 */
360   { "movb",     AL, Ob },
361   { "movS",     eAX, Ov },
362   { "movb",     Ob, AL },
363   { "movS",     Ov, eAX },
364   { "movsb",    Yb, Xb },
365   { "movsS",    Yv, Xv },
366   { "cmpsb",    Xb, Yb },
367   { "cmpsS",    Xv, Yv },
368   /* a8 */
369   { "testb",    AL, Ib },
370   { "testS",    eAX, Iv },
371   { "stosb",    Yb, AL },
372   { "stosS",    Yv, eAX },
373   { "lodsb",    AL, Xb },
374   { "lodsS",    eAX, Xv },
375   { "scasb",    AL, Yb },
376   { "scasS",    eAX, Yv },
377   /* b0 */
378   { "movb",     AL, Ib },
379   { "movb",     CL, Ib },
380   { "movb",     DL, Ib },
381   { "movb",     BL, Ib },
382   { "movb",     AH, Ib },
383   { "movb",     CH, Ib },
384   { "movb",     DH, Ib },
385   { "movb",     BH, Ib },
386   /* b8 */
387   { "movS",     eAX, Iv },
388   { "movS",     eCX, Iv },
389   { "movS",     eDX, Iv },
390   { "movS",     eBX, Iv },
391   { "movS",     eSP, Iv },
392   { "movS",     eBP, Iv },
393   { "movS",     eSI, Iv },
394   { "movS",     eDI, Iv },
395   /* c0 */
396   { GRP2b },
397   { GRP2S },
398   { "ret",      Iw },
399   { "ret" },
400   { "lesS",     Gv, Mp },
401   { "ldsS",     Gv, Mp },
402   { "movb",     Eb, Ib },
403   { "movS",     Ev, Iv },
404   /* c8 */
405   { "enter",    Iw, Ib },
406   { "leave" },
407   { "lret",     Iw },
408   { "lret" },
409   { "int3" },
410   { "int",      Ib },
411   { "into" },
412   { "iret" },
413   /* d0 */
414   { GRP2b_one },
415   { GRP2S_one },
416   { GRP2b_cl },
417   { GRP2S_cl },
418   { "aam",      Ib },
419   { "aad",      Ib },
420   { "(bad)" },
421   { "xlat" },
422   /* d8 */
423   { FLOAT },
424   { FLOAT },
425   { FLOAT },
426   { FLOAT },
427   { FLOAT },
428   { FLOAT },
429   { FLOAT },
430   { FLOAT },
431   /* e0 */
432   { "loopne",   Jb },
433   { "loope",    Jb },
434   { "loop",     Jb },
435   { "jCcxz",    Jb },
436   { "inb",      AL, Ib },
437   { "inS",      eAX, Ib },
438   { "outb",     Ib, AL },
439   { "outS",     Ib, eAX },
440   /* e8 */
441   { "call",     Av },
442   { "jmp",      Jv },
443   { "ljmp",     Ap },
444   { "jmp",      Jb },
445   { "inb",      AL, indirDX },
446   { "inS",      eAX, indirDX },
447   { "outb",     indirDX, AL },
448   { "outS",     indirDX, eAX },
449   /* f0 */
450   { "(bad)" },                  /* lock prefix */
451   { "(bad)" },
452   { "(bad)" },                  /* repne */
453   { "(bad)" },                  /* repz */
454   { "hlt" },
455   { "cmc" },
456   { GRP3b },
457   { GRP3S },
458   /* f8 */
459   { "clc" },
460   { "stc" },
461   { "cli" },
462   { "sti" },
463   { "cld" },
464   { "std" },
465   { GRP4 },
466   { GRP5 },
467 };
468
469 struct dis386 dis386_twobyte[] = {
470   /* 00 */
471   { GRP6 },
472   { GRP7 },
473   { "larS", Gv, Ew },
474   { "lslS", Gv, Ew },  
475   { "(bad)" },
476   { "(bad)" },
477   { "clts" },
478   { "(bad)" },  
479   /* 08 */
480   { "invd" },   { "wbinvd" }, { "(bad)" },  { "(bad)" },  
481   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
482   /* 10 */
483   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
484   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
485   /* 18 */
486   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
487   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
488   /* 20 */
489   /* these are all backward in appendix A of the intel book */
490   { "movl", Rd, Cd },
491   { "movl", Rd, Dd },
492   { "movl", Cd, Rd },
493   { "movl", Dd, Rd },  
494   { "movl", Rd, Td },
495   { "(bad)" },
496   { "movl", Td, Rd },
497   { "(bad)" },  
498   /* 28 */
499   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
500   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
501   /* 30 */
502   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
503   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
504   /* 38 */
505   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
506   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
507   /* 40 */
508   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
509   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
510   /* 48 */
511   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
512   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
513   /* 50 */
514   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
515   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
516   /* 58 */
517   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
518   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
519   /* 60 */
520   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
521   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
522   /* 68 */
523   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
524   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
525   /* 70 */
526   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
527   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
528   /* 78 */
529   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
530   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
531   /* 80 */
532   { "jo", Jv },
533   { "jno", Jv },
534   { "jb", Jv },
535   { "jae", Jv },  
536   { "je", Jv },
537   { "jne", Jv },
538   { "jbe", Jv },
539   { "ja", Jv },  
540   /* 88 */
541   { "js", Jv },
542   { "jns", Jv },
543   { "jp", Jv },
544   { "jnp", Jv },  
545   { "jl", Jv },
546   { "jge", Jv },
547   { "jle", Jv },
548   { "jg", Jv },  
549   /* 90 */
550   { "seto", Eb },
551   { "setno", Eb },
552   { "setb", Eb },
553   { "setae", Eb },
554   { "sete", Eb },
555   { "setne", Eb },
556   { "setbe", Eb },
557   { "seta", Eb },
558   /* 98 */
559   { "sets", Eb },
560   { "setns", Eb },
561   { "setp", Eb },
562   { "setnp", Eb },
563   { "setl", Eb },
564   { "setge", Eb },
565   { "setle", Eb },
566   { "setg", Eb },  
567   /* a0 */
568   { "pushl", fs },
569   { "popl", fs },
570   { "(bad)" },
571   { "btS", Ev, Gv },  
572   { "shldS", Ev, Gv, Ib },
573   { "shldS", Ev, Gv, CL },
574   { "(bad)" },
575   { "(bad)" },  
576   /* a8 */
577   { "pushl", gs },
578   { "popl", gs },
579   { "(bad)" },
580   { "btsS", Ev, Gv },  
581   { "shrdS", Ev, Gv, Ib },
582   { "shrdS", Ev, Gv, CL },
583   { "(bad)" },
584   { "imulS", Gv, Ev },  
585   /* b0 */
586   { "(bad)" },
587   { "(bad)" },
588   { "lssS", Gv, Mp },   /* 386 lists only Mp */
589   { "btrS", Ev, Gv },  
590   { "lfsS", Gv, Mp },   /* 386 lists only Mp */
591   { "lgsS", Gv, Mp },   /* 386 lists only Mp */
592   { "movzbS", Gv, Eb },
593   { "movzwS", Gv, Ew },  
594   /* b8 */
595   { "(bad)" },
596   { "(bad)" },
597   { GRP8 },
598   { "btcS", Ev, Gv },  
599   { "bsfS", Gv, Ev },
600   { "bsrS", Gv, Ev },
601   { "movsbS", Gv, Eb },
602   { "movswS", Gv, Ew },  
603   /* c0 */
604   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
605   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
606   /* c8 */
607   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
608   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
609   /* d0 */
610   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
611   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
612   /* d8 */
613   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
614   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
615   /* e0 */
616   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
617   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
618   /* e8 */
619   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
620   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
621   /* f0 */
622   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
623   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
624   /* f8 */
625   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
626   { "(bad)" },  { "(bad)" },  { "(bad)" },  { "(bad)" },  
627 };
628
629 static char obuf[100];
630 static char *obufp;
631 static char scratchbuf[100];
632 static unsigned char *start_codep;
633 static unsigned char *codep;
634 static int mod;
635 static int rm;
636 static int reg;
637
638 static char *names32[]={
639   "%eax","%ecx","%edx","%ebx", "%esp","%ebp","%esi","%edi",
640 };
641 static char *names16[] = {
642   "%ax","%cx","%dx","%bx","%sp","%bp","%si","%di",
643 };
644 static char *names8[] = {
645   "%al","%cl","%dl","%bl","%ah","%ch","%dh","%bh",
646 };
647 static char *names_seg[] = {
648   "%es","%cs","%ss","%ds","%fs","%gs","%?","%?",
649 };
650 static char *names16_pairs[] = {
651   "%bx+%si","%bx+%di","%bp+%si","%bp+%di","%si","%di","%bp","%bx",
652 };
653
654 struct dis386 grps[][8] = {
655   /* GRP1b */
656   {
657     { "addb",   Eb, Ib },
658     { "orb",    Eb, Ib },
659     { "adcb",   Eb, Ib },
660     { "sbbb",   Eb, Ib },
661     { "andb",   Eb, Ib },
662     { "subb",   Eb, Ib },
663     { "xorb",   Eb, Ib },
664     { "cmpb",   Eb, Ib }
665   },
666   /* GRP1S */
667   {
668     { "addS",   Ev, Iv },
669     { "orS",    Ev, Iv },
670     { "adcS",   Ev, Iv },
671     { "sbbS",   Ev, Iv },
672     { "andS",   Ev, Iv },
673     { "subS",   Ev, Iv },
674     { "xorS",   Ev, Iv },
675     { "cmpS",   Ev, Iv }
676   },
677   /* GRP1Ss */
678   {
679     { "addS",   Ev, sIb },
680     { "orS",    Ev, sIb },
681     { "adcS",   Ev, sIb },
682     { "sbbS",   Ev, sIb },
683     { "andS",   Ev, sIb },
684     { "subS",   Ev, sIb },
685     { "xorS",   Ev, sIb },
686     { "cmpS",   Ev, sIb }
687   },
688   /* GRP2b */
689   {
690     { "rolb",   Eb, Ib },
691     { "rorb",   Eb, Ib },
692     { "rclb",   Eb, Ib },
693     { "rcrb",   Eb, Ib },
694     { "shlb",   Eb, Ib },
695     { "shrb",   Eb, Ib },
696     { "(bad)" },
697     { "sarb",   Eb, Ib },
698   },
699   /* GRP2S */
700   {
701     { "rolS",   Ev, Ib },
702     { "rorS",   Ev, Ib },
703     { "rclS",   Ev, Ib },
704     { "rcrS",   Ev, Ib },
705     { "shlS",   Ev, Ib },
706     { "shrS",   Ev, Ib },
707     { "(bad)" },
708     { "sarS",   Ev, Ib },
709   },
710   /* GRP2b_one */
711   {
712     { "rolb",   Eb },
713     { "rorb",   Eb },
714     { "rclb",   Eb },
715     { "rcrb",   Eb },
716     { "shlb",   Eb },
717     { "shrb",   Eb },
718     { "(bad)" },
719     { "sarb",   Eb },
720   },
721   /* GRP2S_one */
722   {
723     { "rolS",   Ev },
724     { "rorS",   Ev },
725     { "rclS",   Ev },
726     { "rcrS",   Ev },
727     { "shlS",   Ev },
728     { "shrS",   Ev },
729     { "(bad)" },
730     { "sarS",   Ev },
731   },
732   /* GRP2b_cl */
733   {
734     { "rolb",   Eb, CL },
735     { "rorb",   Eb, CL },
736     { "rclb",   Eb, CL },
737     { "rcrb",   Eb, CL },
738     { "shlb",   Eb, CL },
739     { "shrb",   Eb, CL },
740     { "(bad)" },
741     { "sarb",   Eb, CL },
742   },
743   /* GRP2S_cl */
744   {
745     { "rolS",   Ev, CL },
746     { "rorS",   Ev, CL },
747     { "rclS",   Ev, CL },
748     { "rcrS",   Ev, CL },
749     { "shlS",   Ev, CL },
750     { "shrS",   Ev, CL },
751     { "(bad)" },
752     { "sarS",   Ev, CL }
753   },
754   /* GRP3b */
755   {
756     { "testb",  Eb, Ib },
757     { "(bad)",  Eb },
758     { "notb",   Eb },
759     { "negb",   Eb },
760     { "mulb",   AL, Eb },
761     { "imulb",  AL, Eb },
762     { "divb",   AL, Eb },
763     { "idivb",  AL, Eb }
764   },
765   /* GRP3S */
766   {
767     { "testS",  Ev, Iv },
768     { "(bad)" },
769     { "notS",   Ev },
770     { "negS",   Ev },
771     { "mulS",   eAX, Ev },
772     { "imulS",  eAX, Ev },
773     { "divS",   eAX, Ev },
774     { "idivS",  eAX, Ev },
775   },
776   /* GRP4 */
777   {
778     { "incb", Eb },
779     { "decb", Eb },
780     { "(bad)" },
781     { "(bad)" },
782     { "(bad)" },
783     { "(bad)" },
784     { "(bad)" },
785     { "(bad)" },
786   },
787   /* GRP5 */
788   {
789     { "incS",   Ev },
790     { "decS",   Ev },
791     { "call",   indirEv },
792     { "lcall",  indirEv },
793     { "jmp",    indirEv },
794     { "ljmp",   indirEv },
795     { "pushS",  Ev },
796     { "(bad)" },
797   },
798   /* GRP6 */
799   {
800     { "sldt",   Ew },
801     { "str",    Ew },
802     { "lldt",   Ew },
803     { "ltr",    Ew },
804     { "verr",   Ew },
805     { "verw",   Ew },
806     { "(bad)" },
807     { "(bad)" }
808   },
809   /* GRP7 */
810   {
811     { "sgdt", Ew },
812     { "sidt", Ew },
813     { "lgdt", Ew },
814     { "lidt", Ew },
815     { "smsw", Ew },
816     { "(bad)" },
817     { "lmsw", Ew },
818     { "(bad)" },
819   },
820   /* GRP8 */
821   {
822     { "(bad)" },
823     { "(bad)" },
824     { "(bad)" },
825     { "(bad)" },
826     { "btS",    Ev, Ib },
827     { "btsS",   Ev, Ib },
828     { "btrS",   Ev, Ib },
829     { "btcS",   Ev, Ib },
830   }
831 };
832
833 #define PREFIX_REPZ     0x01
834 #define PREFIX_REPNZ    0x02
835 #define PREFIX_LOCK     0x04
836 #define PREFIX_CS       0x08
837 #define PREFIX_SS       0x10
838 #define PREFIX_DS       0x20
839 #define PREFIX_ES       0x40
840 #define PREFIX_FS       0x80
841 #define PREFIX_GS       0x100
842 #define PREFIX_DATA     0x200
843 #define PREFIX_ADR      0x400
844 #define PREFIX_FWAIT    0x800
845
846 static int prefixes;
847
848 ckprefix ()
849 {
850   prefixes = 0;
851   while (1)
852     {
853       switch (*codep)
854         {
855         case 0xf3:
856           prefixes |= PREFIX_REPZ;
857           break;
858         case 0xf2:
859           prefixes |= PREFIX_REPNZ;
860           break;
861         case 0xf0:
862           prefixes |= PREFIX_LOCK;
863           break;
864         case 0x2e:
865           prefixes |= PREFIX_CS;
866           break;
867         case 0x36:
868           prefixes |= PREFIX_SS;
869           break;
870         case 0x3e:
871           prefixes |= PREFIX_DS;
872           break;
873         case 0x26:
874           prefixes |= PREFIX_ES;
875           break;
876         case 0x64:
877           prefixes |= PREFIX_FS;
878           break;
879         case 0x65:
880           prefixes |= PREFIX_GS;
881           break;
882         case 0x66:
883           prefixes |= PREFIX_DATA;
884           break;
885         case 0x67:
886           prefixes |= PREFIX_ADR;
887           break;
888         case 0x9b:
889           prefixes |= PREFIX_FWAIT;
890           break;
891         default:
892           return;
893         }
894       codep++;
895     }
896 }
897
898 static int dflag;
899 static int aflag;               
900
901 static char op1out[100], op2out[100], op3out[100];
902 static unsigned long start_pc;
903
904 /*
905  * disassemble the first instruction in 'inbuf'.  You have to make
906  *   sure all of the bytes of the instruction are filled in.
907  *   On the 386's of 1988, the maximum length of an instruction is 15 bytes.
908  *   (see topic "Redundant prefixes" in the "Differences from 8086"
909  *   section of the "Virtual 8086 Mode" chapter.)
910  * 'pc' should be the address of this instruction, it will
911  *   be used to print the target address if this is a relative jump or call
912  * 'outbuf' gets filled in with the disassembled instruction.  it should
913  *   be long enough to hold the longest disassembled instruction.
914  *   100 bytes is certainly enough, unless symbol printing is added later
915  * The function returns the length of this instruction in bytes.
916  */
917 i386dis (ucs, uip, inbuf, outbuf, mode)
918      unsigned short ucs;
919      unsigned short uip;
920      unsigned char *inbuf;
921      char *outbuf;
922 {
923   struct dis386 *dp;
924   char *p;
925   int i;
926   int enter_instruction;
927   char *first, *second, *third;
928   int needcomma;
929   
930   obuf[0] = 0;
931   op1out[0] = 0;
932   op2out[0] = 0;
933   op3out[0] = 0;
934   
935   start_pc = ucs << 16 | uip;
936   start_codep = inbuf;
937   codep = inbuf;
938   
939   ckprefix ();
940   
941   if (*codep == 0xc8)
942     enter_instruction = 1;
943   else
944     enter_instruction = 0;
945   
946   obufp = obuf;
947   
948   if (prefixes & PREFIX_REPZ)
949     oappend ("repz ");
950   if (prefixes & PREFIX_REPNZ)
951     oappend ("repnz ");
952   if (prefixes & PREFIX_LOCK)
953     oappend ("lock ");
954   
955   if ((prefixes & PREFIX_FWAIT)
956       && ((*codep < 0xd8) || (*codep > 0xdf)))
957     {
958       /* fwait not followed by floating point instruction */
959       oappend ("fwait");
960       strcpy (outbuf, obuf);
961       return (1);
962     }
963   
964   /* these would be initialized to 0 if disassembling for 8086 or 286 */
965     /* these would be initialized to 0 if disassembling for 8086 or 286 */
966   if (mode) {
967     dflag = 1;
968     aflag = 1;
969   } else {
970     dflag = 0;
971     aflag = 0;
972   }
973   
974   if (prefixes & PREFIX_DATA)
975     dflag ^= 1;
976   
977   if (prefixes & PREFIX_ADR)
978     {
979       aflag ^= 1;
980       oappend ("addr16 ");
981     }
982   
983   if (*codep == 0x0f)
984     dp = &dis386_twobyte[*++codep];
985   else
986     dp = &dis386[*codep];
987   codep++;
988   mod = (*codep >> 6) & 3;
989   reg = (*codep >> 3) & 7;
990   rm = *codep & 7;
991   
992   if (dp->name == NULL && dp->bytemode1 == FLOATCODE)
993     {
994       dofloat ();
995     }
996   else
997     {
998       if (dp->name == NULL)
999         dp = &grps[dp->bytemode1][reg];
1000       
1001       putop (dp->name);
1002       
1003       obufp = op1out;
1004       if (dp->op1)
1005         (*dp->op1)(dp->bytemode1);
1006       
1007       obufp = op2out;
1008       if (dp->op2)
1009         (*dp->op2)(dp->bytemode2);
1010       
1011       obufp = op3out;
1012       if (dp->op3)
1013         (*dp->op3)(dp->bytemode3);
1014     }
1015   
1016   obufp = obuf + strlen (obuf);
1017   for (i = strlen (obuf); i < 6; i++)
1018     oappend (" ");
1019   oappend (" ");
1020   
1021   /* enter instruction is printed with operands in the
1022    * same order as the intel book; everything else
1023    * is printed in reverse order 
1024    */
1025   if (enter_instruction)
1026     {
1027       first = op1out;
1028       second = op2out;
1029       third = op3out;
1030     }
1031   else
1032     {
1033       first = op3out;
1034       second = op2out;
1035       third = op1out;
1036     }
1037   needcomma = 0;
1038   if (*first)
1039     {
1040       oappend (first);
1041       needcomma = 1;
1042     }
1043   if (*second)
1044     {
1045       if (needcomma)
1046         oappend (",");
1047       oappend (second);
1048       needcomma = 1;
1049     }
1050   if (*third)
1051     {
1052       if (needcomma)
1053         oappend (",");
1054       oappend (third);
1055     }
1056   strcpy (outbuf, obuf);
1057   return (codep - inbuf);
1058 }
1059
1060 char *float_mem[] = {
1061   /* d8 */
1062   "fadds",
1063   "fmuls",
1064   "fcoms",
1065   "fcomps",
1066   "fsubs",
1067   "fsubrs",
1068   "fdivs",
1069   "fdivrs",
1070   /*  d9 */
1071   "flds",
1072   "(bad)",
1073   "fsts",
1074   "fstps",
1075   "fldenv",
1076   "fldcw",
1077   "fNstenv",
1078   "fNstcw",
1079   /* da */
1080   "fiaddl",
1081   "fimull",
1082   "ficoml",
1083   "ficompl",
1084   "fisubl",
1085   "fisubrl",
1086   "fidivl",
1087   "fidivrl",
1088   /* db */
1089   "fildl",
1090   "(bad)",
1091   "fistl",
1092   "fistpl",
1093   "(bad)",
1094   "fldt",
1095   "(bad)",
1096   "fstpt",
1097   /* dc */
1098   "faddl",
1099   "fmull",
1100   "fcoml",
1101   "fcompl",
1102   "fsubl",
1103   "fsubrl",
1104   "fdivl",
1105   "fdivrl",
1106   /* dd */
1107   "fldl",
1108   "(bad)",
1109   "fstl",
1110   "fstpl",
1111   "frstor",
1112   "(bad)",
1113   "fNsave",
1114   "fNstsw",
1115   /* de */
1116   "fiadd",
1117   "fimul",
1118   "ficom",
1119   "ficomp",
1120   "fisub",
1121   "fisubr",
1122   "fidiv",
1123   "fidivr",
1124   /* df */
1125   "fild",
1126   "(bad)",
1127   "fist",
1128   "fistp",
1129   "fbld",
1130   "fildll",
1131   "fbstp",
1132   "fistpll",
1133 };
1134
1135 #define ST OP_ST, 0
1136 #define STi OP_STi, 0
1137 int OP_ST(), OP_STi();
1138
1139 #define FGRPd9_2 NULL, NULL, 0
1140 #define FGRPd9_4 NULL, NULL, 1
1141 #define FGRPd9_5 NULL, NULL, 2
1142 #define FGRPd9_6 NULL, NULL, 3
1143 #define FGRPd9_7 NULL, NULL, 4
1144 #define FGRPda_5 NULL, NULL, 5
1145 #define FGRPdb_4 NULL, NULL, 6
1146 #define FGRPde_3 NULL, NULL, 7
1147 #define FGRPdf_4 NULL, NULL, 8
1148
1149 struct dis386 float_reg[][8] = {
1150   /* d8 */
1151   {
1152     { "fadd",   ST, STi },
1153     { "fmul",   ST, STi },
1154     { "fcom",   STi },
1155     { "fcomp",  STi },
1156     { "fsub",   ST, STi },
1157     { "fsubr",  ST, STi },
1158     { "fdiv",   ST, STi },
1159     { "fdivr",  ST, STi },
1160   },
1161   /* d9 */
1162   {
1163     { "fld",    STi },
1164     { "fxch",   STi },
1165     { FGRPd9_2 },
1166     { "(bad)" },
1167     { FGRPd9_4 },
1168     { FGRPd9_5 },
1169     { FGRPd9_6 },
1170     { FGRPd9_7 },
1171   },
1172   /* da */
1173   {
1174     { "(bad)" },
1175     { "(bad)" },
1176     { "(bad)" },
1177     { "(bad)" },
1178     { "(bad)" },
1179     { FGRPda_5 },
1180     { "(bad)" },
1181     { "(bad)" },
1182   },
1183   /* db */
1184   {
1185     { "(bad)" },
1186     { "(bad)" },
1187     { "(bad)" },
1188     { "(bad)" },
1189     { FGRPdb_4 },
1190     { "(bad)" },
1191     { "(bad)" },
1192     { "(bad)" },
1193   },
1194   /* dc */
1195   {
1196     { "fadd",   STi, ST },
1197     { "fmul",   STi, ST },
1198     { "(bad)" },
1199     { "(bad)" },
1200     { "fsub",   STi, ST },
1201     { "fsubr",  STi, ST },
1202     { "fdiv",   STi, ST },
1203     { "fdivr",  STi, ST },
1204   },
1205   /* dd */
1206   {
1207     { "ffree",  STi },
1208     { "(bad)" },
1209     { "fst",    STi },
1210     { "fstp",   STi },
1211     { "fucom",  STi },
1212     { "fucomp", STi },
1213     { "(bad)" },
1214     { "(bad)" },
1215   },
1216   /* de */
1217   {
1218     { "faddp",  STi, ST },
1219     { "fmulp",  STi, ST },
1220     { "(bad)" },
1221     { FGRPde_3 },
1222     { "fsubp",  STi, ST },
1223     { "fsubrp", STi, ST },
1224     { "fdivp",  STi, ST },
1225     { "fdivrp", STi, ST },
1226   },
1227   /* df */
1228   {
1229     { "(bad)" },
1230     { "(bad)" },
1231     { "(bad)" },
1232     { "(bad)" },
1233     { FGRPdf_4 },
1234     { "(bad)" },
1235     { "(bad)" },
1236     { "(bad)" },
1237   },
1238 };
1239
1240
1241 char *fgrps[][8] = {
1242   /* d9_2  0 */
1243   {
1244     "fnop","(bad)","(bad)","(bad)","(bad)","(bad)","(bad)","(bad)",
1245   },
1246
1247   /* d9_4  1 */
1248   {
1249     "fchs","fabs","(bad)","(bad)","ftst","fxam","(bad)","(bad)",
1250   },
1251
1252   /* d9_5  2 */
1253   {
1254     "fld1","fldl2t","fldl2e","fldpi","fldlg2","fldln2","fldz","(bad)",
1255   },
1256
1257   /* d9_6  3 */
1258   {
1259     "f2xm1","fyl2x","fptan","fpatan","fxtract","fprem1","fdecstp","fincstp",
1260   },
1261
1262   /* d9_7  4 */
1263   {
1264     "fprem","fyl2xp1","fsqrt","fsincos","frndint","fscale","fsin","fcos",
1265   },
1266
1267   /* da_5  5 */
1268   {
1269     "(bad)","fucompp","(bad)","(bad)","(bad)","(bad)","(bad)","(bad)",
1270   },
1271
1272   /* db_4  6 */
1273   {
1274     "feni(287 only)","fdisi(287 only)","fNclex","fNinit",
1275     "fNsetpm(287 only)","(bad)","(bad)","(bad)",
1276   },
1277
1278   /* de_3  7 */
1279   {
1280     "(bad)","fcompp","(bad)","(bad)","(bad)","(bad)","(bad)","(bad)",
1281   },
1282
1283   /* df_4  8 */
1284   {
1285     "fNstsw","(bad)","(bad)","(bad)","(bad)","(bad)","(bad)","(bad)",
1286   },
1287 };
1288
1289
1290 dofloat ()
1291 {
1292   struct dis386 *dp;
1293   unsigned char floatop;
1294   
1295   floatop = codep[-1];
1296   
1297   if (mod != 3)
1298     {
1299       putop (float_mem[(floatop - 0xd8) * 8 + reg]);
1300       obufp = op1out;
1301       OP_E (v_mode);
1302       return;
1303     }
1304   codep++;
1305   
1306   dp = &float_reg[floatop - 0xd8][reg];
1307   if (dp->name == NULL)
1308     {
1309       putop (fgrps[dp->bytemode1][rm]);
1310       /* instruction fnstsw is only one with strange arg */
1311       if (floatop == 0xdf && *codep == 0xe0)
1312         strcpy (op1out, "%eax");
1313     }
1314   else
1315     {
1316       putop (dp->name);
1317       obufp = op1out;
1318       if (dp->op1)
1319         (*dp->op1)(dp->bytemode1);
1320       obufp = op2out;
1321       if (dp->op2)
1322         (*dp->op2)(dp->bytemode2);
1323     }
1324 }
1325
1326 /* ARGSUSED */
1327 OP_ST (ignore)
1328 {
1329   oappend ("%st");
1330 }
1331
1332 /* ARGSUSED */
1333 OP_STi (ignore)
1334 {
1335   sprintf (scratchbuf, "%%st(%d)", rm);
1336   oappend (scratchbuf);
1337 }
1338
1339
1340 /* capital letters in template are macros */
1341 putop (template)
1342      char *template;
1343 {
1344   char *p;
1345   
1346   for (p = template; *p; p++)
1347     {
1348       switch (*p)
1349         {
1350         default:
1351           *obufp++ = *p;
1352           break;
1353         case 'C':               /* For jcxz/jecxz */
1354           if (aflag == 0)
1355             *obufp++ = 'e';
1356           break;
1357         case 'N':
1358           if ((prefixes & PREFIX_FWAIT) == 0)
1359             *obufp++ = 'n';
1360           break;
1361         case 'S':
1362           /* operand size flag */
1363           if (dflag)
1364             *obufp++ = 'l';
1365           else
1366             *obufp++ = 'w';
1367           break;
1368         }
1369     }
1370   *obufp = 0;
1371 }
1372
1373 oappend (s)
1374 char *s;
1375 {
1376   strcpy (obufp, s);
1377   obufp += strlen (s);
1378   *obufp = 0;
1379 }
1380
1381 append_prefix ()
1382 {
1383   if (prefixes & PREFIX_CS)
1384     oappend ("%cs:");
1385   if (prefixes & PREFIX_DS)
1386     oappend ("%ds:");
1387   if (prefixes & PREFIX_SS)
1388     oappend ("%ss:");
1389   if (prefixes & PREFIX_ES)
1390     oappend ("%es:");
1391   if (prefixes & PREFIX_FS)
1392     oappend ("%fs:");
1393   if (prefixes & PREFIX_GS)
1394     oappend ("%gs:");
1395 }
1396
1397 OP_indirE (bytemode)
1398 {
1399   oappend ("*");
1400   OP_E (bytemode);
1401 }
1402
1403 OP_E (bytemode)
1404 {
1405   int disp;
1406   int havesib;
1407   int didoutput = 0;
1408   int base;
1409   int index;
1410   int scale;
1411   int havebase;
1412   
1413   /* skip mod/rm byte */
1414   codep++;
1415   
1416   havesib = 0;
1417   havebase = 0;
1418   disp = 0;
1419   
1420   if (mod == 3) {
1421       switch (bytemode) {
1422       case b_mode:
1423           oappend (names8[rm]);
1424           break;
1425       case w_mode:
1426           oappend (names16[rm]);
1427           break;
1428       case v_mode:
1429           if (dflag)
1430             oappend (names32[rm]);
1431           else
1432             oappend (names16[rm]);
1433           break;
1434       default:
1435           oappend ("<bad dis table>");
1436           break;
1437       }
1438       return;
1439   }
1440   
1441   append_prefix ();
1442
1443   if (aflag && rm == 4) {
1444       havesib = 1;
1445       havebase = 1;
1446       scale = (*codep >> 6) & 3;
1447       index = (*codep >> 3) & 7;
1448       base = *codep & 7;
1449       codep++;
1450   }
1451   
1452   switch (mod) {
1453   case 0:
1454       if (aflag) {
1455           switch (rm) {
1456           case 4:
1457               /* implies havesib and havebase */
1458               if (base == 5) {
1459                 havebase = 0;
1460                 disp = get32 ();
1461               }
1462               break;
1463           case 5:
1464               disp = get32 ();
1465               break;
1466           default:
1467               havebase = 1;
1468               base = rm;
1469               break;
1470           }
1471       } else {
1472           if (rm == 6) {
1473             havebase = 0;
1474             disp = get16 ();
1475           } else {
1476             havebase = 1;
1477             base = rm;
1478           }
1479       }
1480       break;
1481   case 1:
1482       disp = *(char *)codep++;
1483       if (!aflag || rm != 4) {
1484           havebase = 1;
1485           base = rm;
1486       }
1487       break;
1488   case 2:
1489       if (aflag)
1490         disp = get32 ();
1491       else
1492         disp = get16 ();
1493       if (!aflag || rm != 4) {
1494           havebase = 1;
1495           base = rm;
1496       }
1497       break;
1498   }
1499   
1500   if (mod != 0 || (aflag && rm == 5 || (havesib && base == 5))
1501                || (!aflag && rm == 6)) {
1502     sprintf (scratchbuf, "0x%x", disp);
1503     oappend (scratchbuf);
1504   }
1505   
1506   if (havebase || havesib) {
1507       oappend ("(");
1508       if (havebase)
1509         oappend (aflag ? names32[base] : names16_pairs[base]);
1510       if (havesib) {
1511           if (index != 4) {
1512               sprintf (scratchbuf, ",%s", names32[index]);
1513               oappend (scratchbuf);
1514           }
1515           sprintf (scratchbuf, ",%d", 1 << scale);
1516           oappend (scratchbuf);
1517       }
1518       oappend (")");
1519   }
1520 }
1521
1522 OP_G (bytemode)
1523 {
1524   switch (bytemode) 
1525     {
1526     case b_mode:
1527       oappend (names8[reg]);
1528       break;
1529     case w_mode:
1530       oappend (names16[reg]);
1531       break;
1532     case d_mode:
1533       oappend (names32[reg]);
1534       break;
1535     case v_mode:
1536       if (dflag)
1537         oappend (names32[reg]);
1538       else
1539         oappend (names16[reg]);
1540       break;
1541     default:
1542       oappend ("<internal disassembler error>");
1543       break;
1544     }
1545 }
1546
1547 get32 ()
1548 {
1549   int x = 0;
1550   
1551   x = *codep++ & 0xff;
1552   x |= (*codep++ & 0xff) << 8;
1553   x |= (*codep++ & 0xff) << 16;
1554   x |= (*codep++ & 0xff) << 24;
1555   return (x);
1556 }
1557
1558 get16 ()
1559 {
1560   int x = 0;
1561   
1562   x = *codep++ & 0xff;
1563   x |= (*codep++ & 0xff) << 8;
1564   return (x);
1565 }
1566
1567 OP_REG (code)
1568 {
1569   char *s;
1570   
1571   switch (code) 
1572     {
1573         case indir_dx_reg: s = "(%dx)"; break;
1574         case ax_reg: case cx_reg: case dx_reg: case bx_reg:
1575         case sp_reg: case bp_reg: case si_reg: case di_reg:
1576                 s = names16[code - ax_reg];
1577                 break;
1578         case es_reg: case ss_reg: case cs_reg:
1579         case ds_reg: case fs_reg: case gs_reg:
1580                 s = names_seg[code - es_reg];
1581                 break;
1582         case al_reg: case ah_reg: case cl_reg: case ch_reg:
1583         case dl_reg: case dh_reg: case bl_reg: case bh_reg:
1584                 s = names8[code - al_reg];
1585                 break;
1586         case eAX_reg: case eCX_reg: case eDX_reg: case eBX_reg:
1587         case eSP_reg: case eBP_reg: case eSI_reg: case eDI_reg:
1588       if (dflag)
1589         s = names32[code - eAX_reg];
1590       else
1591         s = names16[code - eAX_reg];
1592       break;
1593     default:
1594       s = "<internal disassembler error>";
1595       break;
1596     }
1597   oappend (s);
1598 }
1599
1600 OP_I (bytemode)
1601 {
1602   int op;
1603   
1604   switch (bytemode) 
1605     {
1606     case b_mode:
1607       op = *codep++ & 0xff;
1608       break;
1609     case v_mode:
1610       if (dflag)
1611         op = get32 ();
1612       else
1613         op = get16 ();
1614       break;
1615     case w_mode:
1616       op = get16 ();
1617       break;
1618     default:
1619       oappend ("<internal disassembler error>");
1620       return;
1621     }
1622   sprintf (scratchbuf, "$0x%x", op);
1623   oappend (scratchbuf);
1624 }
1625
1626 OP_sI (bytemode)
1627 {
1628   int op;
1629   
1630   switch (bytemode) 
1631     {
1632     case b_mode:
1633       op = *(char *)codep++;
1634       break;
1635     case v_mode:
1636       if (dflag)
1637         op = get32 ();
1638       else
1639         op = (short)get16();
1640       break;
1641     case w_mode:
1642       op = (short)get16 ();
1643       break;
1644     default:
1645       oappend ("<internal disassembler error>");
1646       return;
1647     }
1648   sprintf (scratchbuf, "$0x%x", op);
1649   oappend (scratchbuf);
1650 }
1651
1652 OP_J (bytemode)
1653 {
1654   int disp;
1655   
1656   switch (bytemode) 
1657     {
1658     case b_mode:
1659       disp = *(char *)codep++;
1660       append_pc(start_pc + codep - start_codep + disp);
1661       break;
1662     case v_mode:
1663       if (dflag) {
1664         disp = get32 ();
1665         append_pc(start_pc + codep - start_codep + disp);
1666       } else {
1667           disp = (short)get16 ();
1668           disp = (((start_pc + codep - start_codep) & 0xffff) + disp) & 0xffff;
1669           append_pc((start_pc & 0xffff0000) | disp);
1670       }
1671       break;
1672     default:
1673       oappend ("<internal disassembelr error>");
1674       return;
1675     }
1676   
1677   oappend (scratchbuf);
1678 }
1679
1680 append_pc(unsigned long pc)
1681 {
1682   sprintf(scratchbuf, "%04x:%04x", pc >> 16, pc & 0xffff);
1683 }
1684
1685 /* ARGSUSED */
1686 OP_SEG (dummy)
1687 {
1688   static char *sreg[] = {
1689     "%es","%cs","%ss","%ds","%fs","%gs","%?","%?",
1690   };
1691
1692   oappend (sreg[reg]);
1693 }
1694
1695 OP_DIR (size)
1696 {
1697   int seg, offset;
1698   
1699   switch (size) 
1700     {
1701     case lptr:
1702       if (dflag) 
1703         {
1704           offset = get32 ();
1705           seg = get16 ();
1706         } 
1707       else 
1708         {
1709           offset = get16 ();
1710           seg = get16 ();
1711         }
1712       sprintf (scratchbuf, "%04x:%04x", seg, offset);
1713       oappend (scratchbuf);
1714       break;
1715     case v_mode:
1716       if (aflag)
1717         offset = get32 ();
1718       else
1719         offset = (short)get16 ();
1720       
1721       append_pc(start_pc + codep - start_codep + offset);
1722       oappend (scratchbuf);
1723       break;
1724     default:
1725       oappend ("<internal disassembler error>");
1726       break;
1727     }
1728 }
1729
1730 /* ARGSUSED */
1731 OP_OFF (bytemode)
1732 {
1733   int off;
1734   
1735   if (aflag)
1736     off = get32 ();
1737   else
1738     off = get16 ();
1739   
1740   sprintf (scratchbuf, "0x%x", off);
1741   oappend (scratchbuf);
1742 }
1743
1744 /* ARGSUSED */
1745 OP_ESDI (dummy)
1746 {
1747   oappend ("%es:(");
1748   oappend (aflag ? "%edi" : "%di");
1749   oappend (")");
1750 }
1751
1752 /* ARGSUSED */
1753 OP_DSSI (dummy)
1754 {
1755   oappend ("%ds:(");
1756   oappend (aflag ? "%esi" : "%si");
1757   oappend (")");
1758 }
1759
1760 /* ARGSUSED */
1761 OP_ONE (dummy)
1762 {
1763   oappend ("1");
1764 }
1765
1766 /* ARGSUSED */
1767 OP_C (dummy)
1768 {
1769   codep++; /* skip mod/rm */
1770   sprintf (scratchbuf, "%%cr%d", reg);
1771   oappend (scratchbuf);
1772 }
1773
1774 /* ARGSUSED */
1775 OP_D (dummy)
1776 {
1777   codep++; /* skip mod/rm */
1778   sprintf (scratchbuf, "%%db%d", reg);
1779   oappend (scratchbuf);
1780 }
1781
1782 /* ARGSUSED */
1783 OP_T (dummy)
1784 {
1785   codep++; /* skip mod/rm */
1786   sprintf (scratchbuf, "%%tr%d", reg);
1787   oappend (scratchbuf);
1788 }
1789
1790 OP_rm (bytemode)
1791 {
1792   switch (bytemode) 
1793     {
1794     case d_mode:
1795       oappend (names32[rm]);
1796       break;
1797     case w_mode:
1798       oappend (names16[rm]);
1799       break;
1800     }
1801 }
1802
1803 #else
1804
1805 i386dis (pc, inbuf, outbuf, mode)
1806      int pc;
1807      unsigned char *inbuf;
1808      char *outbuf;
1809 {
1810         strcpy (outbuf, "(no disassembler)");
1811         return (1);
1812 }
1813
1814 #endif /* DISASSEMBLER */