VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/IEMAllAImpl.asm@ 37423

Last change on this file since 37423 was 37084, checked in by vboxsync, 14 years ago

IEM: xadd

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 34.4 KB
Line 
1; $Id: IEMAllAImpl.asm 37084 2011-05-13 19:53:02Z vboxsync $
2;; @file
3; IEM - Instruction Implementation in Assembly.
4;
5
6; Copyright (C) 2011 Oracle Corporation
7;
8; This file is part of VirtualBox Open Source Edition (OSE), as
9; available from http://www.virtualbox.org. This file is free software;
10; you can redistribute it and/or modify it under the terms of the GNU
11; General Public License (GPL) as published by the Free Software
12; Foundation, in version 2 as it comes in the "COPYING" file of the
13; VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14; hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15;
16
17
18;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
19; Header Files ;
20;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
21%include "VBox/asmdefs.mac"
22%include "VBox/err.mac"
23%include "VBox/x86.mac"
24
25
26;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
27; Defined Constants And Macros ;
28;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
29
30;
31; We employ some macro assembly here to hid the calling convention differences.
32;
33%ifdef RT_ARCH_AMD64
34 %macro PROLOGUE_1_ARGS 0
35 %endmacro
36 %macro EPILOGUE_1_ARGS 0
37 %endmacro
38 %macro PROLOGUE_2_ARGS 0
39 %endmacro
40 %macro EPILOGUE_2_ARGS 0
41 %endmacro
42 %macro PROLOGUE_3_ARGS 0
43 %endmacro
44 %macro EPILOGUE_3_ARGS 0
45 %endmacro
46 %macro PROLOGUE_4_ARGS 0
47 %endmacro
48 %macro EPILOGUE_4_ARGS 0
49 %endmacro
50
51 %ifdef ASM_CALL64_GCC
52 %define A0 rdi
53 %define A0_32 edi
54 %define A0_16 di
55 %define A0_8 dil
56
57 %define A1 rsi
58 %define A1_32 esi
59 %define A1_16 si
60 %define A1_8 sil
61
62 %define A2 rdx
63 %define A2_32 edx
64 %define A2_16 dx
65 %define A2_8 dl
66
67 %define A3 rcx
68 %define A3_32 ecx
69 %define A3_16 cx
70 %endif
71
72 %ifdef ASM_CALL64_MSC
73 %define A0 rcx
74 %define A0_32 ecx
75 %define A0_16 cx
76 %define A0_8 cl
77
78 %define A1 rdx
79 %define A1_32 edx
80 %define A1_16 dx
81 %define A1_8 dl
82
83 %define A2 r8
84 %define A2_32 r8d
85 %define A2_16 r8w
86 %define A2_8 r8b
87
88 %define A3 r9
89 %define A3_32 r9d
90 %define A3_16 r9w
91 %endif
92
93 %define T0 rax
94 %define T0_32 eax
95 %define T0_16 ax
96 %define T0_8 al
97
98 %define T1 r11
99 %define T1_32 r11d
100 %define T1_16 r11w
101 %define T1_8 r11b
102
103%else
104 ; x86
105 %macro PROLOGUE_1_ARGS 0
106 push edi
107 %endmacro
108 %macro EPILOGUE_1_ARGS 0
109 pop edi
110 %endmacro
111
112 %macro PROLOGUE_2_ARGS 0
113 push edi
114 %endmacro
115 %macro EPILOGUE_2_ARGS 0
116 pop edi
117 %endmacro
118
119 %macro PROLOGUE_3_ARGS 0
120 push ebx
121 mov ebx, [esp + 4 + 4]
122 push edi
123 %endmacro
124 %macro EPILOGUE_3_ARGS 0
125 pop edi
126 pop ebx
127 %endmacro
128
129 %macro PROLOGUE_4_ARGS 0
130 push ebx
131 push edi
132 push esi
133 mov ebx, [esp + 12 + 4 + 0]
134 mov esi, [esp + 12 + 4 + 4]
135 %endmacro
136 %macro EPILOGUE_4_ARGS 0
137 pop esi
138 pop edi
139 pop ebx
140 %endmacro
141
142 %define A0 ecx
143 %define A0_32 ecx
144 %define A0_16 cx
145 %define A0_8 cl
146
147 %define A1 edx
148 %define A1_32 edx
149 %define A1_16 dx
150 %define A1_8 dl
151
152 %define A2 ebx
153 %define A2_32 ebx
154 %define A2_16 bx
155 %define A2_8 bl
156
157 %define A3 esi
158 %define A3_32 esi
159 %define A3_16 si
160
161 %define T0 eax
162 %define T0_32 eax
163 %define T0_16 ax
164 %define T0_8 al
165
166 %define T1 edi
167 %define T1_32 edi
168 %define T1_16 di
169%endif
170
171
172;;
173; Load the relevant flags from [%1] if there are undefined flags (%3).
174;
175; @remarks Clobbers T0, stack. Changes EFLAGS.
176; @param A2 The register pointing to the flags.
177; @param 1 The parameter (A0..A3) pointing to the eflags.
178; @param 2 The set of modified flags.
179; @param 3 The set of undefined flags.
180;
181%macro IEM_MAYBE_LOAD_FLAGS 3
182 ;%if (%3) != 0
183 pushf ; store current flags
184 mov T0_32, [%1] ; load the guest flags
185 and dword [xSP], ~(%2 | %3) ; mask out the modified and undefined flags
186 and T0_32, (%2 | %3) ; select the modified and undefined flags.
187 or [xSP], T0 ; merge guest flags with host flags.
188 popf ; load the mixed flags.
189 ;%endif
190%endmacro
191
192;;
193; Update the flag.
194;
195; @remarks Clobbers T0, T1, stack.
196; @param 1 The register pointing to the EFLAGS.
197; @param 2 The mask of modified flags to save.
198; @param 3 The mask of undefined flags to (maybe) save.
199;
200%macro IEM_SAVE_FLAGS 3
201 %if (%2 | %3) != 0
202 pushf
203 pop T1
204 mov T0_32, [%1] ; flags
205 and T0_32, ~(%2 | %3) ; clear the modified & undefined flags.
206 and T1_32, (%2 | %3) ; select the modified and undefined flags.
207 or T0_32, T1_32 ; combine the flags.
208 mov [%1], T0_32 ; save the flags.
209 %endif
210%endmacro
211
212
213;;
214; Macro for implementing a binary operator.
215;
216; This will generate code for the 8, 16, 32 and 64 bit accesses with locked
217; variants, except on 32-bit system where the 64-bit accesses requires hand
218; coding.
219;
220; All the functions takes a pointer to the destination memory operand in A0,
221; the source register operand in A1 and a pointer to eflags in A2.
222;
223; @param 1 The instruction mnemonic.
224; @param 2 Non-zero if there should be a locked version.
225; @param 3 The modified flags.
226; @param 4 The undefined flags.
227;
228%macro IEMIMPL_BIN_OP 4
229BEGINCODE
230BEGINPROC iemAImpl_ %+ %1 %+ _u8
231 PROLOGUE_3_ARGS
232 IEM_MAYBE_LOAD_FLAGS A2, %3, %4
233 %1 byte [A0], A1_8
234 IEM_SAVE_FLAGS A2, %3, %4
235 EPILOGUE_3_ARGS
236 ret
237ENDPROC iemAImpl_ %+ %1 %+ _u8
238
239BEGINPROC iemAImpl_ %+ %1 %+ _u16
240 PROLOGUE_3_ARGS
241 IEM_MAYBE_LOAD_FLAGS A2, %3, %4
242 %1 word [A0], A1_16
243 IEM_SAVE_FLAGS A2, %3, %4
244 EPILOGUE_3_ARGS
245 ret
246ENDPROC iemAImpl_ %+ %1 %+ _u16
247
248BEGINPROC iemAImpl_ %+ %1 %+ _u32
249 PROLOGUE_3_ARGS
250 IEM_MAYBE_LOAD_FLAGS A2, %3, %4
251 %1 dword [A0], A1_32
252 IEM_SAVE_FLAGS A2, %3, %4
253 EPILOGUE_3_ARGS
254 ret
255ENDPROC iemAImpl_ %+ %1 %+ _u32
256
257 %ifdef RT_ARCH_AMD64
258BEGINPROC iemAImpl_ %+ %1 %+ _u64
259 PROLOGUE_3_ARGS
260 IEM_MAYBE_LOAD_FLAGS A2, %3, %4
261 %1 qword [A0], A1
262 IEM_SAVE_FLAGS A2, %3, %4
263 EPILOGUE_3_ARGS
264 ret
265ENDPROC iemAImpl_ %+ %1 %+ _u64
266 %else ; stub it for now - later, replace with hand coded stuff.
267BEGINPROC iemAImpl_ %+ %1 %+ _u64
268 int3
269 ret
270ENDPROC iemAImpl_ %+ %1 %+ _u64
271 %endif ; !RT_ARCH_AMD64
272
273 %if %2 != 0 ; locked versions requested?
274
275BEGINPROC iemAImpl_ %+ %1 %+ _u8_locked
276 PROLOGUE_3_ARGS
277 IEM_MAYBE_LOAD_FLAGS A2, %3, %4
278 lock %1 byte [A0], A1_8
279 IEM_SAVE_FLAGS A2, %3, %4
280 EPILOGUE_3_ARGS
281 ret
282ENDPROC iemAImpl_ %+ %1 %+ _u8_locked
283
284BEGINPROC iemAImpl_ %+ %1 %+ _u16_locked
285 PROLOGUE_3_ARGS
286 IEM_MAYBE_LOAD_FLAGS A2, %3, %4
287 lock %1 word [A0], A1_16
288 IEM_SAVE_FLAGS A2, %3, %4
289 EPILOGUE_3_ARGS
290 ret
291ENDPROC iemAImpl_ %+ %1 %+ _u16_locked
292
293BEGINPROC iemAImpl_ %+ %1 %+ _u32_locked
294 PROLOGUE_3_ARGS
295 IEM_MAYBE_LOAD_FLAGS A2, %3, %4
296 lock %1 dword [A0], A1_32
297 IEM_SAVE_FLAGS A2, %3, %4
298 EPILOGUE_3_ARGS
299 ret
300ENDPROC iemAImpl_ %+ %1 %+ _u32_locked
301
302 %ifdef RT_ARCH_AMD64
303BEGINPROC iemAImpl_ %+ %1 %+ _u64_locked
304 PROLOGUE_3_ARGS
305 IEM_MAYBE_LOAD_FLAGS A2, %3, %4
306 lock %1 qword [A0], A1
307 IEM_SAVE_FLAGS A2, %3, %4
308 EPILOGUE_3_ARGS
309 ret
310ENDPROC iemAImpl_ %+ %1 %+ _u64_locked
311 %else ; stub it for now - later, replace with hand coded stuff.
312BEGINPROC iemAImpl_ %+ %1 %+ _u64_locked
313 int3
314 ret
315ENDPROC iemAImpl_ %+ %1 %+ _u64_locked
316 %endif ; !RT_ARCH_AMD64
317 %endif ; locked
318%endmacro
319
320; instr,lock,modified-flags.
321IEMIMPL_BIN_OP add, 1, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
322IEMIMPL_BIN_OP adc, 1, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
323IEMIMPL_BIN_OP sub, 1, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
324IEMIMPL_BIN_OP sbb, 1, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
325IEMIMPL_BIN_OP or, 1, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_PF | X86_EFL_CF), X86_EFL_AF,
326IEMIMPL_BIN_OP xor, 1, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_PF | X86_EFL_CF), X86_EFL_AF,
327IEMIMPL_BIN_OP and, 1, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_PF | X86_EFL_CF), X86_EFL_AF,
328IEMIMPL_BIN_OP cmp, 0, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
329IEMIMPL_BIN_OP test, 0, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_PF | X86_EFL_CF), X86_EFL_AF,
330
331
332;;
333; Macro for implementing a bit operator.
334;
335; This will generate code for the 16, 32 and 64 bit accesses with locked
336; variants, except on 32-bit system where the 64-bit accesses requires hand
337; coding.
338;
339; All the functions takes a pointer to the destination memory operand in A0,
340; the source register operand in A1 and a pointer to eflags in A2.
341;
342; @param 1 The instruction mnemonic.
343; @param 2 Non-zero if there should be a locked version.
344; @param 3 The modified flags.
345; @param 4 The undefined flags.
346;
347%macro IEMIMPL_BIT_OP 4
348BEGINCODE
349BEGINPROC iemAImpl_ %+ %1 %+ _u16
350 PROLOGUE_3_ARGS
351 IEM_MAYBE_LOAD_FLAGS A2, %3, %4
352 %1 word [A0], A1_16
353 IEM_SAVE_FLAGS A2, %3, %4
354 EPILOGUE_3_ARGS
355 ret
356ENDPROC iemAImpl_ %+ %1 %+ _u16
357
358BEGINPROC iemAImpl_ %+ %1 %+ _u32
359 PROLOGUE_3_ARGS
360 IEM_MAYBE_LOAD_FLAGS A2, %3, %4
361 %1 dword [A0], A1_32
362 IEM_SAVE_FLAGS A2, %3, %4
363 EPILOGUE_3_ARGS
364 ret
365ENDPROC iemAImpl_ %+ %1 %+ _u32
366
367 %ifdef RT_ARCH_AMD64
368BEGINPROC iemAImpl_ %+ %1 %+ _u64
369 PROLOGUE_3_ARGS
370 IEM_MAYBE_LOAD_FLAGS A2, %3, %4
371 %1 qword [A0], A1
372 IEM_SAVE_FLAGS A2, %3, %4
373 EPILOGUE_3_ARGS
374 ret
375ENDPROC iemAImpl_ %+ %1 %+ _u64
376 %else ; stub it for now - later, replace with hand coded stuff.
377BEGINPROC iemAImpl_ %+ %1 %+ _u64
378 int3
379 ret
380ENDPROC iemAImpl_ %+ %1 %+ _u64
381 %endif ; !RT_ARCH_AMD64
382
383 %if %2 != 0 ; locked versions requested?
384
385BEGINPROC iemAImpl_ %+ %1 %+ _u16_locked
386 PROLOGUE_3_ARGS
387 IEM_MAYBE_LOAD_FLAGS A2, %3, %4
388 lock %1 word [A0], A1_16
389 IEM_SAVE_FLAGS A2, %3, %4
390 EPILOGUE_3_ARGS
391 ret
392ENDPROC iemAImpl_ %+ %1 %+ _u16_locked
393
394BEGINPROC iemAImpl_ %+ %1 %+ _u32_locked
395 PROLOGUE_3_ARGS
396 IEM_MAYBE_LOAD_FLAGS A2, %3, %4
397 lock %1 dword [A0], A1_32
398 IEM_SAVE_FLAGS A2, %3, %4
399 EPILOGUE_3_ARGS
400 ret
401ENDPROC iemAImpl_ %+ %1 %+ _u32_locked
402
403 %ifdef RT_ARCH_AMD64
404BEGINPROC iemAImpl_ %+ %1 %+ _u64_locked
405 PROLOGUE_3_ARGS
406 IEM_MAYBE_LOAD_FLAGS A2, %3, %4
407 lock %1 qword [A0], A1
408 IEM_SAVE_FLAGS A2, %3, %4
409 EPILOGUE_3_ARGS
410 ret
411ENDPROC iemAImpl_ %+ %1 %+ _u64_locked
412 %else ; stub it for now - later, replace with hand coded stuff.
413BEGINPROC iemAImpl_ %+ %1 %+ _u64_locked
414 int3
415 ret
416ENDPROC iemAImpl_ %+ %1 %+ _u64_locked
417 %endif ; !RT_ARCH_AMD64
418 %endif ; locked
419%endmacro
420IEMIMPL_BIT_OP bt, 0, (X86_EFL_CF), (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF)
421IEMIMPL_BIT_OP btc, 1, (X86_EFL_CF), (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF)
422IEMIMPL_BIT_OP bts, 1, (X86_EFL_CF), (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF)
423IEMIMPL_BIT_OP btr, 1, (X86_EFL_CF), (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF)
424
425;;
426; Macro for implementing a bit search operator.
427;
428; This will generate code for the 16, 32 and 64 bit accesses, except on 32-bit
429; system where the 64-bit accesses requires hand coding.
430;
431; All the functions takes a pointer to the destination memory operand in A0,
432; the source register operand in A1 and a pointer to eflags in A2.
433;
434; @param 1 The instruction mnemonic.
435; @param 2 The modified flags.
436; @param 3 The undefined flags.
437;
438%macro IEMIMPL_BIT_OP 3
439BEGINCODE
440BEGINPROC iemAImpl_ %+ %1 %+ _u16
441 PROLOGUE_3_ARGS
442 IEM_MAYBE_LOAD_FLAGS A2, %2, %3
443 %1 T0_16, A1_16
444 mov [A0], T0_16
445 IEM_SAVE_FLAGS A2, %2, %3
446 EPILOGUE_3_ARGS
447 ret
448ENDPROC iemAImpl_ %+ %1 %+ _u16
449
450BEGINPROC iemAImpl_ %+ %1 %+ _u32
451 PROLOGUE_3_ARGS
452 IEM_MAYBE_LOAD_FLAGS A2, %2, %3
453 %1 T0_32, A1_32
454 mov [A0], T0_32
455 IEM_SAVE_FLAGS A2, %2, %3
456 EPILOGUE_3_ARGS
457 ret
458ENDPROC iemAImpl_ %+ %1 %+ _u32
459
460 %ifdef RT_ARCH_AMD64
461BEGINPROC iemAImpl_ %+ %1 %+ _u64
462 PROLOGUE_3_ARGS
463 IEM_MAYBE_LOAD_FLAGS A2, %2, %3
464 %1 T0, A1
465 mov [A0], T0
466 IEM_SAVE_FLAGS A2, %2, %3
467 EPILOGUE_3_ARGS
468 ret
469ENDPROC iemAImpl_ %+ %1 %+ _u64
470 %else ; stub it for now - later, replace with hand coded stuff.
471BEGINPROC iemAImpl_ %+ %1 %+ _u64
472 int3
473 ret
474ENDPROC iemAImpl_ %+ %1 %+ _u64
475 %endif ; !RT_ARCH_AMD64
476%endmacro
477IEMIMPL_BIT_OP bsf, (X86_EFL_ZF), (X86_EFL_OF | X86_EFL_SF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF)
478IEMIMPL_BIT_OP bsr, (X86_EFL_ZF), (X86_EFL_OF | X86_EFL_SF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF)
479
480
481;
482; IMUL is also a similar but yet different case (no lock, no mem dst).
483; The rDX:rAX variant of imul is handled together with mul further down.
484;
485BEGINCODE
486BEGINPROC iemAImpl_imul_two_u16
487 PROLOGUE_3_ARGS
488 IEM_MAYBE_LOAD_FLAGS A2, (X86_EFL_OF | X86_EFL_CF), (X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF)
489 imul A1_16, word [A0]
490 mov [A0], A1_16
491 IEM_SAVE_FLAGS A2, (X86_EFL_OF | X86_EFL_CF), (X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF)
492 EPILOGUE_3_ARGS
493 ret
494ENDPROC iemAImpl_imul_two_u16
495
496BEGINPROC iemAImpl_imul_two_u32
497 PROLOGUE_3_ARGS
498 IEM_MAYBE_LOAD_FLAGS A2, (X86_EFL_OF | X86_EFL_CF), (X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF)
499 imul A1_32, dword [A0]
500 mov [A0], A1_32
501 IEM_SAVE_FLAGS A2, (X86_EFL_OF | X86_EFL_CF), (X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF)
502 EPILOGUE_3_ARGS
503 ret
504ENDPROC iemAImpl_imul_two_u32
505
506BEGINPROC iemAImpl_imul_two_u64
507 PROLOGUE_3_ARGS
508%ifdef RT_ARCH_AMD64
509 IEM_MAYBE_LOAD_FLAGS A2, (X86_EFL_OF | X86_EFL_CF), (X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF)
510 imul A1, qword [A0]
511 mov [A0], A1
512 IEM_SAVE_FLAGS A2, (X86_EFL_OF | X86_EFL_CF), (X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF)
513%else
514 int3 ;; @todo implement me
515%endif
516 EPILOGUE_3_ARGS
517 ret
518ENDPROC iemAImpl_imul_two_u64
519
520
521;
522; XCHG for memory operands. This implies locking. No flag changes.
523;
524; Each function takes two arguments, first the pointer to the memory,
525; then the pointer to the register. They all return void.
526;
527BEGINCODE
528BEGINPROC iemAImpl_xchg_u8
529 PROLOGUE_2_ARGS
530 mov T0_8, [A1]
531 xchg [A0], T0_8
532 mov [A1], T0_8
533 EPILOGUE_2_ARGS
534 ret
535ENDPROC iemAImpl_xchg_u8
536
537BEGINPROC iemAImpl_xchg_u16
538 PROLOGUE_2_ARGS
539 mov T0_16, [A1]
540 xchg [A0], T0_16
541 mov [A1], T0_16
542 EPILOGUE_2_ARGS
543 ret
544ENDPROC iemAImpl_xchg_u16
545
546BEGINPROC iemAImpl_xchg_u32
547 PROLOGUE_2_ARGS
548 mov T0_32, [A1]
549 xchg [A0], T0_32
550 mov [A1], T0_32
551 EPILOGUE_2_ARGS
552 ret
553ENDPROC iemAImpl_xchg_u32
554
555BEGINPROC iemAImpl_xchg_u64
556%ifdef RT_ARCH_AMD64
557 PROLOGUE_2_ARGS
558 mov T0, [A1]
559 xchg [A0], T0
560 mov [A1], T0
561 EPILOGUE_2_ARGS
562 ret
563%else
564 int3
565%endif
566ENDPROC iemAImpl_xchg_u64
567
568
569;
570; XADD for memory operands.
571;
572; Each function takes three arguments, first the pointer to the
573; memory/register, then the pointer to the register, and finally a pointer to
574; eflags. They all return void.
575;
576BEGINCODE
577BEGINPROC iemAImpl_xadd_u8
578 PROLOGUE_3_ARGS
579 IEM_MAYBE_LOAD_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
580 mov T0_8, [A1]
581 xadd [A0], T0_8
582 mov [A1], T0_8
583 IEM_SAVE_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
584 EPILOGUE_3_ARGS
585 ret
586ENDPROC iemAImpl_xadd_u8
587
588BEGINPROC iemAImpl_xadd_u16
589 PROLOGUE_3_ARGS
590 IEM_MAYBE_LOAD_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
591 mov T0_16, [A1]
592 xadd [A0], T0_16
593 mov [A1], T0_16
594 IEM_SAVE_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
595 EPILOGUE_3_ARGS
596 ret
597ENDPROC iemAImpl_xadd_u16
598
599BEGINPROC iemAImpl_xadd_u32
600 PROLOGUE_3_ARGS
601 IEM_MAYBE_LOAD_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
602 mov T0_32, [A1]
603 xadd [A0], T0_32
604 mov [A1], T0_32
605 IEM_SAVE_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
606 EPILOGUE_3_ARGS
607 ret
608ENDPROC iemAImpl_xadd_u32
609
610BEGINPROC iemAImpl_xadd_u64
611%ifdef RT_ARCH_AMD64
612 PROLOGUE_3_ARGS
613 IEM_MAYBE_LOAD_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
614 mov T0, [A1]
615 xadd [A0], T0
616 mov [A1], T0
617 IEM_SAVE_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
618 EPILOGUE_3_ARGS
619 ret
620%else
621 int3
622%endif
623ENDPROC iemAImpl_xadd_u64
624
625BEGINPROC iemAImpl_xadd_u8_locked
626 PROLOGUE_3_ARGS
627 IEM_MAYBE_LOAD_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
628 mov T0_8, [A1]
629 lock xadd [A0], T0_8
630 mov [A1], T0_8
631 IEM_SAVE_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
632 EPILOGUE_3_ARGS
633 ret
634ENDPROC iemAImpl_xadd_u8_locked
635
636BEGINPROC iemAImpl_xadd_u16_locked
637 PROLOGUE_3_ARGS
638 IEM_MAYBE_LOAD_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
639 mov T0_16, [A1]
640 lock xadd [A0], T0_16
641 mov [A1], T0_16
642 IEM_SAVE_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
643 EPILOGUE_3_ARGS
644 ret
645ENDPROC iemAImpl_xadd_u16_locked
646
647BEGINPROC iemAImpl_xadd_u32_locked
648 PROLOGUE_3_ARGS
649 IEM_MAYBE_LOAD_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
650 mov T0_32, [A1]
651 lock xadd [A0], T0_32
652 mov [A1], T0_32
653 IEM_SAVE_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
654 EPILOGUE_3_ARGS
655 ret
656ENDPROC iemAImpl_xadd_u32_locked
657
658BEGINPROC iemAImpl_xadd_u64_locked
659%ifdef RT_ARCH_AMD64
660 PROLOGUE_3_ARGS
661 IEM_MAYBE_LOAD_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
662 mov T0, [A1]
663 lock xadd [A0], T0
664 mov [A1], T0
665 IEM_SAVE_FLAGS A2, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
666 EPILOGUE_3_ARGS
667 ret
668%else
669 int3
670%endif
671ENDPROC iemAImpl_xadd_u64_locked
672
673
674;;
675; Macro for implementing a unary operator.
676;
677; This will generate code for the 8, 16, 32 and 64 bit accesses with locked
678; variants, except on 32-bit system where the 64-bit accesses requires hand
679; coding.
680;
681; All the functions takes a pointer to the destination memory operand in A0,
682; the source register operand in A1 and a pointer to eflags in A2.
683;
684; @param 1 The instruction mnemonic.
685; @param 2 The modified flags.
686; @param 3 The undefined flags.
687;
688%macro IEMIMPL_UNARY_OP 3
689BEGINCODE
690BEGINPROC iemAImpl_ %+ %1 %+ _u8
691 PROLOGUE_2_ARGS
692 IEM_MAYBE_LOAD_FLAGS A1, %2, %3
693 %1 byte [A0]
694 IEM_SAVE_FLAGS A1, %2, %3
695 EPILOGUE_2_ARGS
696 ret
697ENDPROC iemAImpl_ %+ %1 %+ _u8
698
699BEGINPROC iemAImpl_ %+ %1 %+ _u8_locked
700 PROLOGUE_2_ARGS
701 IEM_MAYBE_LOAD_FLAGS A1, %2, %3
702 lock %1 byte [A0]
703 IEM_SAVE_FLAGS A1, %2, %3
704 EPILOGUE_2_ARGS
705 ret
706ENDPROC iemAImpl_ %+ %1 %+ _u8_locked
707
708BEGINPROC iemAImpl_ %+ %1 %+ _u16
709 PROLOGUE_2_ARGS
710 IEM_MAYBE_LOAD_FLAGS A1, %2, %3
711 %1 word [A0]
712 IEM_SAVE_FLAGS A1, %2, %3
713 EPILOGUE_2_ARGS
714 ret
715ENDPROC iemAImpl_ %+ %1 %+ _u16
716
717BEGINPROC iemAImpl_ %+ %1 %+ _u16_locked
718 PROLOGUE_2_ARGS
719 IEM_MAYBE_LOAD_FLAGS A1, %2, %3
720 lock %1 word [A0]
721 IEM_SAVE_FLAGS A1, %2, %3
722 EPILOGUE_2_ARGS
723 ret
724ENDPROC iemAImpl_ %+ %1 %+ _u16_locked
725
726BEGINPROC iemAImpl_ %+ %1 %+ _u32
727 PROLOGUE_2_ARGS
728 IEM_MAYBE_LOAD_FLAGS A1, %2, %3
729 %1 dword [A0]
730 IEM_SAVE_FLAGS A1, %2, %3
731 EPILOGUE_2_ARGS
732 ret
733ENDPROC iemAImpl_ %+ %1 %+ _u32
734
735BEGINPROC iemAImpl_ %+ %1 %+ _u32_locked
736 PROLOGUE_2_ARGS
737 IEM_MAYBE_LOAD_FLAGS A1, %2, %3
738 lock %1 dword [A0]
739 IEM_SAVE_FLAGS A1, %2, %3
740 EPILOGUE_2_ARGS
741 ret
742ENDPROC iemAImpl_ %+ %1 %+ _u32_locked
743
744 %ifdef RT_ARCH_AMD64
745BEGINPROC iemAImpl_ %+ %1 %+ _u64
746 PROLOGUE_2_ARGS
747 IEM_MAYBE_LOAD_FLAGS A1, %2, %3
748 %1 qword [A0]
749 IEM_SAVE_FLAGS A1, %2, %3
750 EPILOGUE_2_ARGS
751 ret
752ENDPROC iemAImpl_ %+ %1 %+ _u64
753
754BEGINPROC iemAImpl_ %+ %1 %+ _u64_locked
755 PROLOGUE_2_ARGS
756 IEM_MAYBE_LOAD_FLAGS A1, %2, %3
757 lock %1 qword [A0]
758 IEM_SAVE_FLAGS A1, %2, %3
759 EPILOGUE_2_ARGS
760 ret
761ENDPROC iemAImpl_ %+ %1 %+ _u64_locked
762 %else
763 ; stub them for now.
764BEGINPROC iemAImpl_ %+ %1 %+ _u64
765 int3
766 ret
767ENDPROC iemAImpl_ %+ %1 %+ _u64
768BEGINPROC iemAImpl_ %+ %1 %+ _u64_locked
769 int3
770 ret
771ENDPROC iemAImpl_ %+ %1 %+ _u64_locked
772 %endif
773
774%endmacro
775
776IEMIMPL_UNARY_OP inc, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF), 0
777IEMIMPL_UNARY_OP dec, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF), 0
778IEMIMPL_UNARY_OP neg, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF), 0
779IEMIMPL_UNARY_OP not, 0, 0
780
781
782
783;;
784; Macro for implementing a shift operation.
785;
786; This will generate code for the 8, 16, 32 and 64 bit accesses, except on
787; 32-bit system where the 64-bit accesses requires hand coding.
788;
789; All the functions takes a pointer to the destination memory operand in A0,
790; the shift count in A1 and a pointer to eflags in A2.
791;
792; @param 1 The instruction mnemonic.
793; @param 2 The modified flags.
794; @param 3 The undefined flags.
795;
796; Makes ASSUMPTIONS about A0, A1 and A2 assignments.
797;
798%macro IEMIMPL_SHIFT_OP 3
799BEGINCODE
800BEGINPROC iemAImpl_ %+ %1 %+ _u8
801 PROLOGUE_3_ARGS
802 IEM_MAYBE_LOAD_FLAGS A2, %2, %3
803 %ifdef ASM_CALL64_GCC
804 mov cl, A1_8
805 %1 byte [A0], cl
806 %else
807 xchg A1, A0
808 %1 byte [A1], cl
809 %endif
810 IEM_SAVE_FLAGS A2, %2, %3
811 EPILOGUE_3_ARGS
812 ret
813ENDPROC iemAImpl_ %+ %1 %+ _u8
814
815BEGINPROC iemAImpl_ %+ %1 %+ _u16
816 PROLOGUE_3_ARGS
817 IEM_MAYBE_LOAD_FLAGS A2, %2, %3
818 %ifdef ASM_CALL64_GCC
819 mov cl, A1_8
820 %1 word [A0], cl
821 %else
822 xchg A1, A0
823 %1 word [A1], cl
824 %endif
825 IEM_SAVE_FLAGS A2, %2, %3
826 EPILOGUE_3_ARGS
827 ret
828ENDPROC iemAImpl_ %+ %1 %+ _u16
829
830BEGINPROC iemAImpl_ %+ %1 %+ _u32
831 PROLOGUE_3_ARGS
832 IEM_MAYBE_LOAD_FLAGS A2, %2, %3
833 %ifdef ASM_CALL64_GCC
834 mov cl, A1_8
835 %1 dword [A0], cl
836 %else
837 xchg A1, A0
838 %1 dword [A1], cl
839 %endif
840 IEM_SAVE_FLAGS A2, %2, %3
841 EPILOGUE_3_ARGS
842 ret
843ENDPROC iemAImpl_ %+ %1 %+ _u32
844
845 %ifdef RT_ARCH_AMD64
846BEGINPROC iemAImpl_ %+ %1 %+ _u64
847 PROLOGUE_3_ARGS
848 IEM_MAYBE_LOAD_FLAGS A2, %2, %3
849 %ifdef ASM_CALL64_GCC
850 mov cl, A1_8
851 %1 qword [A0], cl
852 %else
853 xchg A1, A0
854 %1 qword [A1], cl
855 %endif
856 IEM_SAVE_FLAGS A2, %2, %3
857 EPILOGUE_3_ARGS
858 ret
859ENDPROC iemAImpl_ %+ %1 %+ _u64
860 %else ; stub it for now - later, replace with hand coded stuff.
861BEGINPROC iemAImpl_ %+ %1 %+ _u64
862 int3
863 ret
864ENDPROC iemAImpl_ %+ %1 %+ _u64
865 %endif ; !RT_ARCH_AMD64
866
867%endmacro
868
869IEMIMPL_SHIFT_OP rol, (X86_EFL_OF | X86_EFL_CF), 0
870IEMIMPL_SHIFT_OP ror, (X86_EFL_OF | X86_EFL_CF), 0
871IEMIMPL_SHIFT_OP rcl, (X86_EFL_OF | X86_EFL_CF), 0
872IEMIMPL_SHIFT_OP rcr, (X86_EFL_OF | X86_EFL_CF), 0
873IEMIMPL_SHIFT_OP shl, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_PF | X86_EFL_CF), (X86_EFL_AF)
874IEMIMPL_SHIFT_OP shr, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_PF | X86_EFL_CF), (X86_EFL_AF)
875IEMIMPL_SHIFT_OP sar, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_PF | X86_EFL_CF), (X86_EFL_AF)
876
877
878;;
879; Macro for implementing a doulbe precision shift operation.
880;
881; This will generate code for the 16, 32 and 64 bit accesses, except on
882; 32-bit system where the 64-bit accesses requires hand coding.
883;
884; The functions takes the destination operand (r/m) in A0, the source (reg) in
885; A1, the shift count in A2 and a pointer to the eflags variable/register in A3.
886;
887; @param 1 The instruction mnemonic.
888; @param 2 The modified flags.
889; @param 3 The undefined flags.
890;
891; Makes ASSUMPTIONS about A0, A1, A2 and A3 assignments.
892;
893%macro IEMIMPL_SHIFT_DBL_OP 3
894BEGINCODE
895BEGINPROC iemAImpl_ %+ %1 %+ _u16
896 PROLOGUE_4_ARGS
897 IEM_MAYBE_LOAD_FLAGS A3, %2, %3
898 %ifdef ASM_CALL64_GCC
899 xchg A3, A2
900 %1 [A0], A1_16, cl
901 xchg A3, A2
902 %else
903 xchg A0, A2
904 %1 [A2], A1_16, cl
905 %endif
906 IEM_SAVE_FLAGS A3, %2, %3
907 EPILOGUE_4_ARGS
908 ret
909ENDPROC iemAImpl_ %+ %1 %+ _u16
910
911BEGINPROC iemAImpl_ %+ %1 %+ _u32
912 PROLOGUE_4_ARGS
913 IEM_MAYBE_LOAD_FLAGS A3, %2, %3
914 %ifdef ASM_CALL64_GCC
915 xchg A3, A2
916 %1 [A0], A1_32, cl
917 xchg A3, A2
918 %else
919 xchg A0, A2
920 %1 [A2], A1_32, cl
921 %endif
922 IEM_SAVE_FLAGS A3, %2, %3
923 EPILOGUE_4_ARGS
924 ret
925ENDPROC iemAImpl_ %+ %1 %+ _u32
926
927 %ifdef RT_ARCH_AMD64
928BEGINPROC iemAImpl_ %+ %1 %+ _u64
929 PROLOGUE_4_ARGS
930 IEM_MAYBE_LOAD_FLAGS A3, %2, %3
931 %ifdef ASM_CALL64_GCC
932 xchg A3, A2
933 %1 [A0], A1, cl
934 xchg A3, A2
935 %else
936 xchg A0, A2
937 %1 [A2], A1, cl
938 %endif
939 IEM_SAVE_FLAGS A3, %2, %3
940 EPILOGUE_4_ARGS
941 ret
942ENDPROC iemAImpl_ %+ %1 %+ _u64
943 %else ; stub it for now - later, replace with hand coded stuff.
944BEGINPROC iemAImpl_ %+ %1 %+ _u64
945 int3
946 ret
947ENDPROC iemAImpl_ %+ %1 %+ _u64
948 %endif ; !RT_ARCH_AMD64
949
950%endmacro
951
952IEMIMPL_SHIFT_DBL_OP shld, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_PF | X86_EFL_CF), (X86_EFL_AF)
953IEMIMPL_SHIFT_DBL_OP shrd, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_PF | X86_EFL_CF), (X86_EFL_AF)
954
955
956;;
957; Macro for implementing a multiplication operations.
958;
959; This will generate code for the 8, 16, 32 and 64 bit accesses, except on
960; 32-bit system where the 64-bit accesses requires hand coding.
961;
962; The 8-bit function only operates on AX, so it takes no DX pointer. The other
963; functions takes a pointer to rAX in A0, rDX in A1, the operand in A2 and a
964; pointer to eflags in A3.
965;
966; The functions all return 0 so the caller can be used for div/idiv as well as
967; for the mul/imul implementation.
968;
969; @param 1 The instruction mnemonic.
970; @param 2 The modified flags.
971; @param 3 The undefined flags.
972;
973; Makes ASSUMPTIONS about A0, A1, A2, A3, T0 and T1 assignments.
974;
975%macro IEMIMPL_MUL_OP 3
976BEGINCODE
977BEGINPROC iemAImpl_ %+ %1 %+ _u8
978 PROLOGUE_3_ARGS
979 IEM_MAYBE_LOAD_FLAGS A2, %2, %3
980 mov al, [A0]
981 %1 A1_8
982 mov [A0], ax
983 IEM_SAVE_FLAGS A2, %2, %3
984 EPILOGUE_3_ARGS
985 xor eax, eax
986 ret
987ENDPROC iemAImpl_ %+ %1 %+ _u8
988
989BEGINPROC iemAImpl_ %+ %1 %+ _u16
990 PROLOGUE_4_ARGS
991 IEM_MAYBE_LOAD_FLAGS A3, %2, %3
992 mov ax, [A0]
993 %ifdef ASM_CALL64_GCC
994 %1 A2_16
995 mov [A0], ax
996 mov [A1], dx
997 %else
998 mov T1, A1
999 %1 A2_16
1000 mov [A0], ax
1001 mov [T1], dx
1002 %endif
1003 IEM_SAVE_FLAGS A3, %2, %3
1004 EPILOGUE_4_ARGS
1005 xor eax, eax
1006 ret
1007ENDPROC iemAImpl_ %+ %1 %+ _u16
1008
1009BEGINPROC iemAImpl_ %+ %1 %+ _u32
1010 PROLOGUE_4_ARGS
1011 IEM_MAYBE_LOAD_FLAGS A3, %2, %3
1012 mov eax, [A0]
1013 %ifdef ASM_CALL64_GCC
1014 %1 A2_32
1015 mov [A0], eax
1016 mov [A1], edx
1017 %else
1018 mov T1, A1
1019 %1 A2_32
1020 mov [A0], eax
1021 mov [T1], edx
1022 %endif
1023 IEM_SAVE_FLAGS A3, %2, %3
1024 EPILOGUE_4_ARGS
1025 xor eax, eax
1026 ret
1027ENDPROC iemAImpl_ %+ %1 %+ _u32
1028
1029 %ifdef RT_ARCH_AMD64
1030BEGINPROC iemAImpl_ %+ %1 %+ _u64
1031 PROLOGUE_4_ARGS
1032 IEM_MAYBE_LOAD_FLAGS A3, %2, %3
1033 mov rax, [A0]
1034 %ifdef ASM_CALL64_GCC
1035 %1 A2
1036 mov [A0], rax
1037 mov [A1], rdx
1038 %else
1039 mov T1, A1
1040 %1 A2
1041 mov [A0], rax
1042 mov [T1], rdx
1043 %endif
1044 IEM_SAVE_FLAGS A3, %2, %3
1045 EPILOGUE_4_ARGS
1046 xor eax, eax
1047 ret
1048ENDPROC iemAImpl_ %+ %1 %+ _u64
1049 %else ; stub it for now - later, replace with hand coded stuff.
1050BEGINPROC iemAImpl_ %+ %1 %+ _u64
1051 int3
1052 ret
1053ENDPROC iemAImpl_ %+ %1 %+ _u64
1054 %endif ; !RT_ARCH_AMD64
1055
1056%endmacro
1057
1058IEMIMPL_MUL_OP mul, (X86_EFL_OF | X86_EFL_CF), (X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF)
1059IEMIMPL_MUL_OP imul, (X86_EFL_OF | X86_EFL_CF), (X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF)
1060
1061
1062;;
1063; Macro for implementing a division operations.
1064;
1065; This will generate code for the 8, 16, 32 and 64 bit accesses, except on
1066; 32-bit system where the 64-bit accesses requires hand coding.
1067;
1068; The 8-bit function only operates on AX, so it takes no DX pointer. The other
1069; functions takes a pointer to rAX in A0, rDX in A1, the operand in A2 and a
1070; pointer to eflags in A3.
1071;
1072; The functions all return 0 on success and -1 if a divide error should be
1073; raised by the caller.
1074;
1075; @param 1 The instruction mnemonic.
1076; @param 2 The modified flags.
1077; @param 3 The undefined flags.
1078;
1079; Makes ASSUMPTIONS about A0, A1, A2, A3, T0 and T1 assignments.
1080;
1081%macro IEMIMPL_DIV_OP 3
1082BEGINCODE
1083BEGINPROC iemAImpl_ %+ %1 %+ _u8
1084 PROLOGUE_3_ARGS
1085
1086 test A1_8, A1_8
1087 jz .div_zero
1088 ;; @todo test for overflow
1089
1090 IEM_MAYBE_LOAD_FLAGS A2, %2, %3
1091 mov ax, [A0]
1092 %1 A1_8
1093 mov [A0], ax
1094 IEM_SAVE_FLAGS A2, %2, %3
1095 xor eax, eax
1096
1097.return:
1098 EPILOGUE_3_ARGS
1099 ret
1100.div_zero:
1101 mov eax, -1
1102 jmp .return
1103ENDPROC iemAImpl_ %+ %1 %+ _u8
1104
1105BEGINPROC iemAImpl_ %+ %1 %+ _u16
1106 PROLOGUE_4_ARGS
1107
1108 test A1_16, A1_16
1109 jz .div_zero
1110 ;; @todo test for overflow
1111
1112 IEM_MAYBE_LOAD_FLAGS A3, %2, %3
1113 %ifdef ASM_CALL64_GCC
1114 mov T1, A2
1115 mov ax, [A0]
1116 mov dx, [A1]
1117 %1 T1_16
1118 mov [A0], ax
1119 mov [A1], dx
1120 %else
1121 mov T1, A1
1122 mov ax, [A0]
1123 mov dx, [T1]
1124 %1 A2_16
1125 mov [A0], ax
1126 mov [T1], dx
1127 %endif
1128 IEM_SAVE_FLAGS A3, %2, %3
1129 xor eax, eax
1130
1131.return:
1132 EPILOGUE_4_ARGS
1133 ret
1134.div_zero:
1135 mov eax, -1
1136 jmp .return
1137ENDPROC iemAImpl_ %+ %1 %+ _u16
1138
1139BEGINPROC iemAImpl_ %+ %1 %+ _u32
1140 PROLOGUE_4_ARGS
1141
1142 test A1_32, A1_32
1143 jz .div_zero
1144 ;; @todo test for overflow
1145
1146 IEM_MAYBE_LOAD_FLAGS A3, %2, %3
1147 mov eax, [A0]
1148 %ifdef ASM_CALL64_GCC
1149 mov T1, A2
1150 mov eax, [A0]
1151 mov edx, [A1]
1152 %1 T1_32
1153 mov [A0], eax
1154 mov [A1], edx
1155 %else
1156 mov T1, A1
1157 mov eax, [A0]
1158 mov edx, [T1]
1159 %1 A2_32
1160 mov [A0], eax
1161 mov [T1], edx
1162 %endif
1163 IEM_SAVE_FLAGS A3, %2, %3
1164 xor eax, eax
1165
1166.return:
1167 EPILOGUE_4_ARGS
1168 ret
1169.div_zero:
1170 mov eax, -1
1171 jmp .return
1172ENDPROC iemAImpl_ %+ %1 %+ _u32
1173
1174 %ifdef RT_ARCH_AMD64
1175BEGINPROC iemAImpl_ %+ %1 %+ _u64
1176 PROLOGUE_4_ARGS
1177
1178 test A1, A1
1179 jz .div_zero
1180 ;; @todo test for overflow
1181
1182 IEM_MAYBE_LOAD_FLAGS A3, %2, %3
1183 mov rax, [A0]
1184 %ifdef ASM_CALL64_GCC
1185 mov T1, A2
1186 mov rax, [A0]
1187 mov rdx, [A1]
1188 %1 T1
1189 mov [A0], rax
1190 mov [A1], rdx
1191 %else
1192 mov T1, A1
1193 mov rax, [A0]
1194 mov rdx, [T1]
1195 %1 A2
1196 mov [A0], rax
1197 mov [T1], rdx
1198 %endif
1199 IEM_SAVE_FLAGS A3, %2, %3
1200 xor eax, eax
1201
1202.return:
1203 EPILOGUE_4_ARGS
1204 ret
1205.div_zero:
1206 mov eax, -1
1207 jmp .return
1208ENDPROC iemAImpl_ %+ %1 %+ _u64
1209 %else ; stub it for now - later, replace with hand coded stuff.
1210BEGINPROC iemAImpl_ %+ %1 %+ _u64
1211 int3
1212 ret
1213ENDPROC iemAImpl_ %+ %1 %+ _u64
1214 %endif ; !RT_ARCH_AMD64
1215
1216%endmacro
1217
1218IEMIMPL_DIV_OP div, 0, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF)
1219IEMIMPL_DIV_OP idiv, 0, (X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF)
1220
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette