VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/EMAllA.asm@ 5429

Last change on this file since 5429 was 5384, checked in by vboxsync, 17 years ago

LOCK BTR and LOCK OR (for Solaris guests).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 24.3 KB
Line 
1; $Id: EMAllA.asm 5384 2007-10-19 14:13:42Z vboxsync $
2;; @file
3; EM Assembly Routines.
4;
5
6;
7; Copyright (C) 2006-2007 innotek GmbH
8;
9; This file is part of VirtualBox Open Source Edition (OSE), as
10; available from http://www.virtualbox.org. This file is free software;
11; you can redistribute it and/or modify it under the terms of the GNU
12; General Public License as published by the Free Software Foundation,
13; in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
14; distribution. VirtualBox OSE is distributed in the hope that it will
15; be useful, but WITHOUT ANY WARRANTY of any kind.
16
17;*******************************************************************************
18;* Header Files *
19;*******************************************************************************
20%include "VBox/asmdefs.mac"
21%include "VBox/err.mac"
22%include "VBox/x86.mac"
23
24;; @def MY_PTR_REG
25; The register we use for value pointers (And,Or,Dec,Inc).
26%ifdef RT_ARCH_AMD64
27%define MY_PTR_REG rcx
28%else
29%define MY_PTR_REG ecx
30%endif
31
32;; @def MY_RET_REG
33; The register we return the result in.
34%ifdef RT_ARCH_AMD64
35%define MY_RET_REG rax
36%else
37%define MY_RET_REG eax
38%endif
39
40BEGINCODE
41
42
43;;
44; Emulate CMP instruction, CDECL calling conv.
45; EMDECL(uint32_t) EMEmulateCmp(uint32_t u32Param1, uint32_t u32Param2, size_t cb);
46;
47; @returns EFLAGS after the operation, only arithmetic flags is valid.
48; @param [esp + 04h] rdi rcx Param 1 - First parameter (Dst).
49; @param [esp + 08h] rsi edx Param 2 - Second parameter (Src).
50; @param [esp + 0ch] rdx r8 Param 3 - Size of parameters, only 1/2/4 is valid.
51;
52align 16
53BEGINPROC EMEmulateCmp
54%ifdef RT_ARCH_AMD64
55%ifdef RT_OS_WINDOWS
56 mov rax, r8 ; eax = size of parameters
57%else ; !RT_OS_WINDOWS
58 mov rax, rdx ; rax = size of parameters
59 mov rcx, rdi ; rcx = first parameter
60 mov rdx, rsi ; rdx = second parameter
61%endif ; !RT_OS_WINDOWS
62%else ; !RT_ARCH_AMD64
63 mov eax, [esp + 0ch] ; eax = size of parameters
64 mov ecx, [esp + 04h] ; ecx = first parameter
65 mov edx, [esp + 08h] ; edx = second parameter
66%endif
67
68 ; switch on size
69%ifdef RT_ARCH_AMD64
70 cmp al, 8
71 je short .do_qword ; 8 bytes variant
72%endif
73 cmp al, 4
74 je short .do_dword ; 4 bytes variant
75 cmp al, 2
76 je short .do_word ; 2 byte variant
77 cmp al, 1
78 je short .do_byte ; 1 bytes variant
79 int3
80
81 ; workers
82%ifdef RT_ARCH_AMD64
83.do_qword:
84 cmp rcx, rdx ; do 8 bytes CMP
85 jmp short .done
86%endif
87
88.do_dword:
89 cmp ecx, edx ; do 4 bytes CMP
90 jmp short .done
91
92.do_word:
93 cmp cx, dx ; do 2 bytes CMP
94 jmp short .done
95
96.do_byte:
97 cmp cl, dl ; do 1 byte CMP
98
99 ; collect flags and return.
100.done:
101 pushf
102 pop MY_RET_REG
103 retn
104ENDPROC EMEmulateCmp
105
106
107;;
108; Emulate AND instruction, CDECL calling conv.
109; EMDECL(uint32_t) EMEmulateAnd(uint32_t *pu32Param1, uint32_t u32Param2, size_t cb);
110;
111; @returns EFLAGS after the operation, only arithmetic flags is valid.
112; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
113; @param [esp + 08h] Param 2 - Second parameter.
114; @param [esp + 0ch] Param 3 - Size of parameters, only 1/2/4 is valid.
115; @uses eax, ecx, edx
116;
117align 16
118BEGINPROC EMEmulateAnd
119%ifdef RT_ARCH_AMD64
120%ifdef RT_OS_WINDOWS
121 mov rax, r8 ; eax = size of parameters
122%else ; !RT_OS_WINDOWS
123 mov rax, rdx ; rax = size of parameters
124 mov rcx, rdi ; rcx = first parameter
125 mov rdx, rsi ; rdx = second parameter
126%endif ; !RT_OS_WINDOWS
127%else ; !RT_ARCH_AMD64
128 mov eax, [esp + 0ch] ; eax = size of parameters
129 mov ecx, [esp + 04h] ; ecx = first parameter
130 mov edx, [esp + 08h] ; edx = second parameter
131%endif
132
133 ; switch on size
134%ifdef RT_ARCH_AMD64
135 cmp al, 8
136 je short .do_qword ; 8 bytes variant
137%endif
138 cmp al, 4
139 je short .do_dword ; 4 bytes variant
140 cmp al, 2
141 je short .do_word ; 2 byte variant
142 cmp al, 1
143 je short .do_byte ; 1 bytes variant
144 int3
145
146 ; workers
147%ifdef RT_ARCH_AMD64
148.do_qword:
149 and [MY_PTR_REG], rdx ; do 8 bytes AND
150 jmp short .done
151%endif
152
153.do_dword:
154 and [MY_PTR_REG], edx ; do 4 bytes AND
155 jmp short .done
156
157.do_word:
158 and [MY_PTR_REG], dx ; do 2 bytes AND
159 jmp short .done
160
161.do_byte:
162 and [MY_PTR_REG], dl ; do 1 byte AND
163
164 ; collect flags and return.
165.done:
166 pushf
167 pop MY_RET_REG
168 retn
169ENDPROC EMEmulateAnd
170
171
172;;
173; Emulate OR instruction, CDECL calling conv.
174; EMDECL(uint32_t) EMEmulateOr(uint32_t *pu32Param1, uint32_t u32Param2, size_t cb);
175;
176; @returns EFLAGS after the operation, only arithmetic flags is valid.
177; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
178; @param [esp + 08h] Param 2 - Second parameter.
179; @param [esp + 0ch] Param 3 - Size of parameters, only 1/2/4 is valid.
180; @uses eax, ecx, edx
181;
182align 16
183BEGINPROC EMEmulateOr
184%ifdef RT_ARCH_AMD64
185%ifdef RT_OS_WINDOWS
186 mov rax, r8 ; eax = size of parameters
187%else ; !RT_OS_WINDOWS
188 mov rax, rdx ; rax = size of parameters
189 mov rcx, rdi ; rcx = first parameter
190 mov rdx, rsi ; rdx = second parameter
191%endif ; !RT_OS_WINDOWS
192%else ; !RT_ARCH_AMD64
193 mov eax, [esp + 0ch] ; eax = size of parameters
194 mov ecx, [esp + 04h] ; ecx = first parameter
195 mov edx, [esp + 08h] ; edx = second parameter
196%endif
197
198 ; switch on size
199%ifdef RT_ARCH_AMD64
200 cmp al, 8
201 je short .do_qword ; 8 bytes variant
202%endif
203 cmp al, 4
204 je short .do_dword ; 4 bytes variant
205 cmp al, 2
206 je short .do_word ; 2 byte variant
207 cmp al, 1
208 je short .do_byte ; 1 bytes variant
209 int3
210
211 ; workers
212%ifdef RT_ARCH_AMD64
213.do_qword:
214 or [MY_PTR_REG], rdx ; do 8 bytes OR
215 jmp short .done
216%endif
217
218.do_dword:
219 or [MY_PTR_REG], edx ; do 4 bytes OR
220 jmp short .done
221
222.do_word:
223 or [MY_PTR_REG], dx ; do 2 bytes OR
224 jmp short .done
225
226.do_byte:
227 or [MY_PTR_REG], dl ; do 1 byte OR
228
229 ; collect flags and return.
230.done:
231 pushf
232 pop MY_RET_REG
233 retn
234ENDPROC EMEmulateOr
235
236;;
237; Emulate LOCK OR instruction.
238; EMDECL(int) EMEmulateLockOr(RTGCPTR GCPtrParam1, RTGCUINTREG Param2, size_t cbSize, RTGCUINTREG *pf);
239;
240; @returns VINF_SUCCESS on success, VERR_ACCESS_DENIED on \#PF (GC only).
241; @param [esp + 04h] gcc:rdi msc:rcx Param 1 - First parameter - pointer to data item (the real stuff).
242; @param [esp + 08h] gcc:rsi msc:rdx Param 2 - Second parameter- the immediate / register value.
243; @param [esp + 0ch] gcc:rdx msc:r8 Param 3 - Size of the operation - 1, 2, 4 or 8 bytes.
244; @param [esp + 10h] gcc:rcx msc:r9 Param 4 - Where to store the eflags on success.
245; only arithmetic flags are valid.
246align 16
247BEGINPROC EMEmulateLockOr
248%ifdef RT_ARCH_AMD64
249%ifdef RT_OS_WINDOWS
250 mov rax, r8 ; eax = size of parameters
251%else ; !RT_OS_WINDOWS
252 mov rax, rdx ; rax = size of parameters
253 mov rcx, rdi ; rcx = first parameter
254 mov rdx, rsi ; rdx = second parameter
255%endif ; !RT_OS_WINDOWS
256%else ; !RT_ARCH_AMD64
257 mov eax, [esp + 0ch] ; eax = size of parameters
258 mov ecx, [esp + 04h] ; ecx = first parameter (MY_PTR_REG)
259 mov edx, [esp + 08h] ; edx = second parameter
260%endif
261
262 ; switch on size
263%ifdef RT_ARCH_AMD64
264 cmp al, 8
265 je short .do_qword ; 8 bytes variant
266%endif
267 cmp al, 4
268 je short .do_dword ; 4 bytes variant
269 cmp al, 2
270 je short .do_word ; 2 byte variant
271 cmp al, 1
272 je short .do_byte ; 1 bytes variant
273 int3
274
275 ; workers
276%ifdef RT_ARCH_AMD64
277.do_qword:
278 lock or [MY_PTR_REG], rdx ; do 8 bytes OR
279 jmp short .done
280%endif
281
282.do_dword:
283 lock or [MY_PTR_REG], edx ; do 4 bytes OR
284 jmp short .done
285
286.do_word:
287 lock or [MY_PTR_REG], dx ; do 2 bytes OR
288 jmp short .done
289
290.do_byte:
291 lock or [MY_PTR_REG], dl ; do 1 byte OR
292
293 ; collect flags and return.
294.done:
295 pushf
296%ifdef RT_ARCH_AMD64
297 pop rax
298 %ifdef RT_OS_WINDOWS
299 mov [r9], eax
300 %else ; !RT_OS_WINDOWS
301 mov [rcx], eax
302 %endif ; !RT_OS_WINDOWS
303%else ; !RT_ARCH_AMD64
304 mov eax, [esp + 10h + 4]
305 pop dword [eax]
306%endif
307 mov eax, VINF_SUCCESS
308 retn
309
310%ifdef IN_GC
311; #PF resume point.
312GLOBALNAME EMEmulateLockOr_Error
313 mov eax, VERR_ACCESS_DENIED
314 ret
315%endif
316
317ENDPROC EMEmulateLockOr
318
319;;
320; Emulate XOR instruction, CDECL calling conv.
321; EMDECL(uint32_t) EMEmulateXor(uint32_t *pu32Param1, uint32_t u32Param2, size_t cb);
322;
323; @returns EFLAGS after the operation, only arithmetic flags is valid.
324; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
325; @param [esp + 08h] Param 2 - Second parameter.
326; @param [esp + 0ch] Param 3 - Size of parameters, only 1/2/4 is valid.
327; @uses eax, ecx, edx
328;
329align 16
330BEGINPROC EMEmulateXor
331%ifdef RT_ARCH_AMD64
332%ifdef RT_OS_WINDOWS
333 mov rax, r8 ; eax = size of parameters
334%else ; !RT_OS_WINDOWS
335 mov rax, rdx ; rax = size of parameters
336 mov rcx, rdi ; rcx = first parameter
337 mov rdx, rsi ; rdx = second parameter
338%endif ; !RT_OS_WINDOWS
339%else ; !RT_ARCH_AMD64
340 mov eax, [esp + 0ch] ; eax = size of parameters
341 mov ecx, [esp + 04h] ; ecx = first parameter
342 mov edx, [esp + 08h] ; edx = second parameter
343%endif
344
345 ; switch on size
346%ifdef RT_ARCH_AMD64
347 cmp al, 8
348 je short .do_qword ; 8 bytes variant
349%endif
350 cmp al, 4
351 je short .do_dword ; 4 bytes variant
352 cmp al, 2
353 je short .do_word ; 2 byte variant
354 cmp al, 1
355 je short .do_byte ; 1 bytes variant
356 int3
357
358 ; workers
359%ifdef RT_ARCH_AMD64
360.do_qword:
361 xor [MY_PTR_REG], rdx ; do 8 bytes XOR
362 jmp short .done
363%endif
364
365.do_dword:
366 xor [MY_PTR_REG], edx ; do 4 bytes XOR
367 jmp short .done
368
369.do_word:
370 xor [MY_PTR_REG], dx ; do 2 bytes XOR
371 jmp short .done
372
373.do_byte:
374 xor [MY_PTR_REG], dl ; do 1 byte XOR
375
376 ; collect flags and return.
377.done:
378 pushf
379 pop MY_RET_REG
380 retn
381ENDPROC EMEmulateXor
382
383;;
384; Emulate INC instruction, CDECL calling conv.
385; EMDECL(uint32_t) EMEmulateInc(uint32_t *pu32Param1, size_t cb);
386;
387; @returns EFLAGS after the operation, only arithmetic flags are valid.
388; @param [esp + 04h] rdi rcx Param 1 - First parameter - pointer to data item.
389; @param [esp + 08h] rsi rdx Param 2 - Size of parameters, only 1/2/4 is valid.
390; @uses eax, ecx, edx
391;
392align 16
393BEGINPROC EMEmulateInc
394%ifdef RT_ARCH_AMD64
395%ifdef RT_OS_WINDOWS
396 mov rax, rdx ; eax = size of parameters
397%else ; !RT_OS_WINDOWS
398 mov rax, rsi ; eax = size of parameters
399 mov rcx, rdi ; rcx = first parameter
400%endif ; !RT_OS_WINDOWS
401%else ; !RT_ARCH_AMD64
402 mov eax, [esp + 08h] ; eax = size of parameters
403 mov ecx, [esp + 04h] ; ecx = first parameter
404%endif
405
406 ; switch on size
407%ifdef RT_ARCH_AMD64
408 cmp al, 8
409 je short .do_qword ; 8 bytes variant
410%endif
411 cmp al, 4
412 je short .do_dword ; 4 bytes variant
413 cmp al, 2
414 je short .do_word ; 2 byte variant
415 cmp al, 1
416 je short .do_byte ; 1 bytes variant
417 int3
418
419 ; workers
420%ifdef RT_ARCH_AMD64
421.do_qword:
422 inc qword [MY_PTR_REG] ; do 8 bytes INC
423 jmp short .done
424%endif
425
426.do_dword:
427 inc dword [MY_PTR_REG] ; do 4 bytes INC
428 jmp short .done
429
430.do_word:
431 inc word [MY_PTR_REG] ; do 2 bytes INC
432 jmp short .done
433
434.do_byte:
435 inc byte [MY_PTR_REG] ; do 1 byte INC
436 jmp short .done
437
438 ; collect flags and return.
439.done:
440 pushf
441 pop MY_RET_REG
442 retn
443ENDPROC EMEmulateInc
444
445
446;;
447; Emulate DEC instruction, CDECL calling conv.
448; EMDECL(uint32_t) EMEmulateDec(uint32_t *pu32Param1, size_t cb);
449;
450; @returns EFLAGS after the operation, only arithmetic flags are valid.
451; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
452; @param [esp + 08h] Param 2 - Size of parameters, only 1/2/4 is valid.
453; @uses eax, ecx, edx
454;
455align 16
456BEGINPROC EMEmulateDec
457%ifdef RT_ARCH_AMD64
458%ifdef RT_OS_WINDOWS
459 mov rax, rdx ; eax = size of parameters
460%else ; !RT_OS_WINDOWS
461 mov rax, rsi ; eax = size of parameters
462 mov rcx, rdi ; rcx = first parameter
463%endif ; !RT_OS_WINDOWS
464%else ; !RT_ARCH_AMD64
465 mov eax, [esp + 08h] ; eax = size of parameters
466 mov ecx, [esp + 04h] ; ecx = first parameter
467%endif
468
469 ; switch on size
470%ifdef RT_ARCH_AMD64
471 cmp al, 8
472 je short .do_qword ; 8 bytes variant
473%endif
474 cmp al, 4
475 je short .do_dword ; 4 bytes variant
476 cmp al, 2
477 je short .do_word ; 2 byte variant
478 cmp al, 1
479 je short .do_byte ; 1 bytes variant
480 int3
481
482 ; workers
483%ifdef RT_ARCH_AMD64
484.do_qword:
485 dec qword [MY_PTR_REG] ; do 8 bytes DEC
486 jmp short .done
487%endif
488
489.do_dword:
490 dec dword [MY_PTR_REG] ; do 4 bytes DEC
491 jmp short .done
492
493.do_word:
494 dec word [MY_PTR_REG] ; do 2 bytes DEC
495 jmp short .done
496
497.do_byte:
498 dec byte [MY_PTR_REG] ; do 1 byte DEC
499 jmp short .done
500
501 ; collect flags and return.
502.done:
503 pushf
504 pop MY_RET_REG
505 retn
506ENDPROC EMEmulateDec
507
508;;
509; Emulate ADD instruction, CDECL calling conv.
510; EMDECL(uint32_t) EMEmulateAdd(uint32_t *pu32Param1, uint32_t u32Param2, size_t cb);
511;
512; @returns EFLAGS after the operation, only arithmetic flags is valid.
513; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
514; @param [esp + 08h] Param 2 - Second parameter.
515; @param [esp + 0ch] Param 3 - Size of parameters, only 1/2/4 is valid.
516; @uses eax, ecx, edx
517;
518align 16
519BEGINPROC EMEmulateAdd
520%ifdef RT_ARCH_AMD64
521%ifdef RT_OS_WINDOWS
522 mov rax, r8 ; eax = size of parameters
523%else ; !RT_OS_WINDOWS
524 mov rax, rdx ; rax = size of parameters
525 mov rcx, rdi ; rcx = first parameter
526 mov rdx, rsi ; rdx = second parameter
527%endif ; !RT_OS_WINDOWS
528%else ; !RT_ARCH_AMD64
529 mov eax, [esp + 0ch] ; eax = size of parameters
530 mov ecx, [esp + 04h] ; ecx = first parameter
531 mov edx, [esp + 08h] ; edx = second parameter
532%endif
533
534 ; switch on size
535%ifdef RT_ARCH_AMD64
536 cmp al, 8
537 je short .do_qword ; 8 bytes variant
538%endif
539 cmp al, 4
540 je short .do_dword ; 4 bytes variant
541 cmp al, 2
542 je short .do_word ; 2 byte variant
543 cmp al, 1
544 je short .do_byte ; 1 bytes variant
545 int3
546
547 ; workers
548%ifdef RT_ARCH_AMD64
549.do_qword:
550 add [MY_PTR_REG], rdx ; do 8 bytes ADD
551 jmp short .done
552%endif
553
554.do_dword:
555 add [MY_PTR_REG], edx ; do 4 bytes ADD
556 jmp short .done
557
558.do_word:
559 add [MY_PTR_REG], dx ; do 2 bytes ADD
560 jmp short .done
561
562.do_byte:
563 add [MY_PTR_REG], dl ; do 1 byte ADD
564
565 ; collect flags and return.
566.done:
567 pushf
568 pop MY_RET_REG
569 retn
570ENDPROC EMEmulateAdd
571
572;;
573; Emulate ADC instruction, CDECL calling conv.
574; EMDECL(uint32_t) EMEmulateAdcWithCarrySet(uint32_t *pu32Param1, uint32_t u32Param2, size_t cb);
575;
576; @returns EFLAGS after the operation, only arithmetic flags is valid.
577; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
578; @param [esp + 08h] Param 2 - Second parameter.
579; @param [esp + 0ch] Param 3 - Size of parameters, only 1/2/4 is valid.
580; @uses eax, ecx, edx
581;
582align 16
583BEGINPROC EMEmulateAdcWithCarrySet
584%ifdef RT_ARCH_AMD64
585%ifdef RT_OS_WINDOWS
586 mov rax, r8 ; eax = size of parameters
587%else ; !RT_OS_WINDOWS
588 mov rax, rdx ; rax = size of parameters
589 mov rcx, rdi ; rcx = first parameter
590 mov rdx, rsi ; rdx = second parameter
591%endif ; !RT_OS_WINDOWS
592%else ; !RT_ARCH_AMD64
593 mov eax, [esp + 0ch] ; eax = size of parameters
594 mov ecx, [esp + 04h] ; ecx = first parameter
595 mov edx, [esp + 08h] ; edx = second parameter
596%endif
597
598 ; switch on size
599%ifdef RT_ARCH_AMD64
600 cmp al, 8
601 je short .do_qword ; 8 bytes variant
602%endif
603 cmp al, 4
604 je short .do_dword ; 4 bytes variant
605 cmp al, 2
606 je short .do_word ; 2 byte variant
607 cmp al, 1
608 je short .do_byte ; 1 bytes variant
609 int3
610
611 ; workers
612%ifdef RT_ARCH_AMD64
613.do_qword:
614 stc ; set carry flag
615 adc [MY_PTR_REG], rdx ; do 8 bytes ADC
616 jmp short .done
617%endif
618
619.do_dword:
620 stc ; set carry flag
621 adc [MY_PTR_REG], edx ; do 4 bytes ADC
622 jmp short .done
623
624.do_word:
625 stc ; set carry flag
626 adc [MY_PTR_REG], dx ; do 2 bytes ADC
627 jmp short .done
628
629.do_byte:
630 stc ; set carry flag
631 adc [MY_PTR_REG], dl ; do 1 byte ADC
632
633 ; collect flags and return.
634.done:
635 pushf
636 pop MY_RET_REG
637 retn
638ENDPROC EMEmulateAdcWithCarrySet
639
640;;
641; Emulate SUB instruction, CDECL calling conv.
642; EMDECL(uint32_t) EMEmulateSub(uint32_t *pu32Param1, uint32_t u32Param2, size_t cb);
643;
644; @returns EFLAGS after the operation, only arithmetic flags is valid.
645; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
646; @param [esp + 08h] Param 2 - Second parameter.
647; @param [esp + 0ch] Param 3 - Size of parameters, only 1/2/4 is valid.
648; @uses eax, ecx, edx
649;
650align 16
651BEGINPROC EMEmulateSub
652%ifdef RT_ARCH_AMD64
653%ifdef RT_OS_WINDOWS
654 mov rax, r8 ; eax = size of parameters
655%else ; !RT_OS_WINDOWS
656 mov rax, rdx ; rax = size of parameters
657 mov rcx, rdi ; rcx = first parameter
658 mov rdx, rsi ; rdx = second parameter
659%endif ; !RT_OS_WINDOWS
660%else ; !RT_ARCH_AMD64
661 mov eax, [esp + 0ch] ; eax = size of parameters
662 mov ecx, [esp + 04h] ; ecx = first parameter
663 mov edx, [esp + 08h] ; edx = second parameter
664%endif
665
666 ; switch on size
667%ifdef RT_ARCH_AMD64
668 cmp al, 8
669 je short .do_qword ; 8 bytes variant
670%endif
671 cmp al, 4
672 je short .do_dword ; 4 bytes variant
673 cmp al, 2
674 je short .do_word ; 2 byte variant
675 cmp al, 1
676 je short .do_byte ; 1 bytes variant
677 int3
678
679 ; workers
680%ifdef RT_ARCH_AMD64
681.do_qword:
682 sub [MY_PTR_REG], rdx ; do 8 bytes SUB
683 jmp short .done
684%endif
685
686.do_dword:
687 sub [MY_PTR_REG], edx ; do 4 bytes SUB
688 jmp short .done
689
690.do_word:
691 sub [MY_PTR_REG], dx ; do 2 bytes SUB
692 jmp short .done
693
694.do_byte:
695 sub [MY_PTR_REG], dl ; do 1 byte SUB
696
697 ; collect flags and return.
698.done:
699 pushf
700 pop MY_RET_REG
701 retn
702ENDPROC EMEmulateSub
703
704
705;;
706; Emulate BTR instruction, CDECL calling conv.
707; EMDECL(uint32_t) EMEmulateBtr(uint32_t *pu32Param1, uint32_t u32Param2);
708;
709; @returns EFLAGS after the operation, only arithmetic flags is valid.
710; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
711; @param [esp + 08h] Param 2 - Second parameter.
712; @uses eax, ecx, edx
713;
714align 16
715BEGINPROC EMEmulateBtr
716%ifdef RT_ARCH_AMD64
717%ifndef RT_OS_WINDOWS
718 mov rcx, rdi ; rcx = first parameter
719 mov rdx, rsi ; rdx = second parameter
720%endif ; !RT_OS_WINDOWS
721%else ; !RT_ARCH_AMD64
722 mov ecx, [esp + 04h] ; ecx = first parameter
723 mov edx, [esp + 08h] ; edx = second parameter
724%endif
725
726 and edx, 7
727 btr [MY_PTR_REG], edx
728
729 ; collect flags and return.
730 pushf
731 pop MY_RET_REG
732 retn
733ENDPROC EMEmulateBtr
734
735;;
736; Emulate LOCK BTR instruction.
737; EMDECL(int) EMEmulateLockBtr(RTGCPTR GCPtrParam1, RTGCUINTREG Param2, uint32_t *pf);
738;
739; @returns VINF_SUCCESS on success, VERR_ACCESS_DENIED on \#PF (GC only).
740; @param [esp + 04h] gcc:rdi msc:rcx Param 1 - First parameter - pointer to data item (the real stuff).
741; @param [esp + 08h] gcc:rsi msc:rdx Param 2 - Second parameter- the immediate / register value. (really an 8 byte value)
742; @param [esp + 0ch] gcc:rdx msc:r8 Param 3 - Where to store the eflags on success.
743;
744align 16
745BEGINPROC EMEmulateLockBtr
746%ifdef RT_ARCH_AMD64
747 %ifdef RT_OS_WINDOWS
748 mov rax, r8 ; rax = third parameter
749 %else ; !RT_OS_WINDOWS
750 mov rcx, rdi ; rcx = first parameter
751 mov rax, rdx ; rax = third parameter
752 mov rdx, rsi ; rdx = second parameter
753 %endif ; !RT_OS_WINDOWS
754%else ; !RT_ARCH_AMD64
755 mov ecx, [esp + 04h] ; ecx = first parameter
756 mov edx, [esp + 08h] ; edx = second parameter
757 mov eax, [esp + 0ch] ; eax = third parameter
758%endif
759
760 lock btr [MY_PTR_REG], edx
761
762 ; collect flags and return.
763 pushf
764 pop xDX
765 mov [xAX], edx
766 mov eax, VINF_SUCCESS
767 retn
768
769%ifdef IN_GC
770; #PF resume point.
771GLOBALNAME EMEmulateLockBtr_Error
772 mov eax, VERR_ACCESS_DENIED
773 ret
774%endif
775
776ENDPROC EMEmulateLockBtr
777
778;;
779; Emulate BTC instruction, CDECL calling conv.
780; EMDECL(uint32_t) EMEmulateBtc(uint32_t *pu32Param1, uint32_t u32Param2);
781;
782; @returns EFLAGS after the operation, only arithmetic flags is valid.
783; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
784; @param [esp + 08h] Param 2 - Second parameter.
785; @uses eax, ecx, edx
786;
787align 16
788BEGINPROC EMEmulateBtc
789%ifdef RT_ARCH_AMD64
790%ifndef RT_OS_WINDOWS
791 mov rcx, rdi ; rcx = first parameter
792 mov rdx, rsi ; rdx = second parameter
793%endif ; !RT_OS_WINDOWS
794%else ; !RT_ARCH_AMD64
795 mov ecx, [esp + 04h] ; ecx = first parameter
796 mov edx, [esp + 08h] ; edx = second parameter
797%endif
798
799 and edx, 7
800 btc [MY_PTR_REG], edx
801
802 ; collect flags and return.
803 pushf
804 pop MY_RET_REG
805 retn
806ENDPROC EMEmulateBtc
807
808;;
809; Emulate BTS instruction, CDECL calling conv.
810; EMDECL(uint32_t) EMEmulateBts(uint32_t *pu32Param1, uint32_t u32Param2);
811;
812; @returns EFLAGS after the operation, only arithmetic flags is valid.
813; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
814; @param [esp + 08h] Param 2 - Second parameter.
815; @uses eax, ecx, edx
816;
817align 16
818BEGINPROC EMEmulateBts
819%ifdef RT_ARCH_AMD64
820%ifndef RT_OS_WINDOWS
821 mov rcx, rdi ; rcx = first parameter
822 mov rdx, rsi ; rdx = second parameter
823%endif ; !RT_OS_WINDOWS
824%else ; !RT_ARCH_AMD64
825 mov ecx, [esp + 04h] ; ecx = first parameter
826 mov edx, [esp + 08h] ; edx = second parameter
827%endif
828
829 and edx, 7
830 bts [MY_PTR_REG], edx
831
832 ; collect flags and return.
833 pushf
834 pop MY_RET_REG
835 retn
836ENDPROC EMEmulateBts
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