VirtualBox

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

Last change on this file since 12446 was 12005, checked in by vboxsync, 16 years ago

Comment update

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 32.5 KB
Line 
1; $Id: EMAllA.asm 12005 2008-09-02 18:25:16Z vboxsync $
2;; @file
3; EM Assembly Routines.
4;
5
6;
7; Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 (GPL) as published by the Free Software
13; Foundation, in version 2 as it comes in the "COPYING" file of the
14; VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15; hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16;
17; Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18; Clara, CA 95054 USA or visit http://www.sun.com if you need
19; additional information or have any questions.
20;
21
22;*******************************************************************************
23;* Header Files *
24;*******************************************************************************
25%include "VBox/asmdefs.mac"
26%include "VBox/err.mac"
27%include "VBox/x86.mac"
28
29;; @def MY_PTR_REG
30; The register we use for value pointers (And,Or,Dec,Inc).
31%ifdef RT_ARCH_AMD64
32%define MY_PTR_REG rcx
33%else
34%define MY_PTR_REG ecx
35%endif
36
37;; @def MY_RET_REG
38; The register we return the result in.
39%ifdef RT_ARCH_AMD64
40%define MY_RET_REG rax
41%else
42%define MY_RET_REG eax
43%endif
44
45BEGINCODE
46
47
48;;
49; Emulate CMP instruction, CDECL calling conv.
50; EMDECL(uint32_t) EMEmulateCmp(uint32_t u32Param1, uint64_t u64Param2, size_t cb);
51;
52; @returns EFLAGS after the operation, only arithmetic flags are valid.
53; @param [esp + 04h] rdi rcx Param 1 - First parameter (Dst).
54; @param [esp + 08h] rsi edx Param 2 - Second parameter (Src).
55; @param [esp + 10h] Param 3 - Size of parameters, only 1/2/4/8 (64 bits host only) is valid.
56;
57align 16
58BEGINPROC EMEmulateCmp
59%ifdef RT_ARCH_AMD64
60%ifdef RT_OS_WINDOWS
61 mov rax, r8 ; eax = size of parameters
62%else ; !RT_OS_WINDOWS
63 mov rax, rdx ; rax = size of parameters
64 mov rcx, rdi ; rcx = first parameter
65 mov rdx, rsi ; rdx = second parameter
66%endif ; !RT_OS_WINDOWS
67%else ; !RT_ARCH_AMD64
68 mov eax, [esp + 10h] ; eax = size of parameters
69 mov ecx, [esp + 04h] ; ecx = first parameter
70 mov edx, [esp + 08h] ; edx = second parameter
71%endif
72
73 ; switch on size
74%ifdef RT_ARCH_AMD64
75 cmp al, 8
76 je short .do_qword ; 8 bytes variant
77%endif
78 cmp al, 4
79 je short .do_dword ; 4 bytes variant
80 cmp al, 2
81 je short .do_word ; 2 byte variant
82 cmp al, 1
83 je short .do_byte ; 1 bytes variant
84 int3
85
86 ; workers
87%ifdef RT_ARCH_AMD64
88.do_qword:
89 cmp rcx, rdx ; do 8 bytes CMP
90 jmp short .done
91%endif
92
93.do_dword:
94 cmp ecx, edx ; do 4 bytes CMP
95 jmp short .done
96
97.do_word:
98 cmp cx, dx ; do 2 bytes CMP
99 jmp short .done
100
101.do_byte:
102 cmp cl, dl ; do 1 byte CMP
103
104 ; collect flags and return.
105.done:
106 pushf
107 pop MY_RET_REG
108 retn
109ENDPROC EMEmulateCmp
110
111
112;;
113; Emulate AND instruction, CDECL calling conv.
114; EMDECL(uint32_t) EMEmulateAnd(void *pvParam1, uint64_t u64Param2, size_t cb);
115;
116; @returns EFLAGS after the operation, only arithmetic flags are valid.
117; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
118; @param [esp + 08h] Param 2 - Second parameter.
119; @param [esp + 10h] Param 3 - Size of parameters, only 1/2/4/8 (64 bits host only) is valid.
120; @uses eax, ecx, edx
121;
122align 16
123BEGINPROC EMEmulateAnd
124%ifdef RT_ARCH_AMD64
125%ifdef RT_OS_WINDOWS
126 mov rax, r8 ; eax = size of parameters
127%else ; !RT_OS_WINDOWS
128 mov rax, rdx ; rax = size of parameters
129 mov rcx, rdi ; rcx = first parameter
130 mov rdx, rsi ; rdx = second parameter
131%endif ; !RT_OS_WINDOWS
132%else ; !RT_ARCH_AMD64
133 mov eax, [esp + 10h] ; eax = size of parameters
134 mov ecx, [esp + 04h] ; ecx = first parameter
135 mov edx, [esp + 08h] ; edx = second parameter
136%endif
137
138 ; switch on size
139%ifdef RT_ARCH_AMD64
140 cmp al, 8
141 je short .do_qword ; 8 bytes variant
142%endif
143 cmp al, 4
144 je short .do_dword ; 4 bytes variant
145 cmp al, 2
146 je short .do_word ; 2 byte variant
147 cmp al, 1
148 je short .do_byte ; 1 bytes variant
149 int3
150
151 ; workers
152%ifdef RT_ARCH_AMD64
153.do_qword:
154 and [MY_PTR_REG], rdx ; do 8 bytes AND
155 jmp short .done
156%endif
157
158.do_dword:
159 and [MY_PTR_REG], edx ; do 4 bytes AND
160 jmp short .done
161
162.do_word:
163 and [MY_PTR_REG], dx ; do 2 bytes AND
164 jmp short .done
165
166.do_byte:
167 and [MY_PTR_REG], dl ; do 1 byte AND
168
169 ; collect flags and return.
170.done:
171 pushf
172 pop MY_RET_REG
173 retn
174ENDPROC EMEmulateAnd
175
176
177;;
178; Emulate OR instruction, CDECL calling conv.
179; EMDECL(uint32_t) EMEmulateOr(void *pvParam1, uint64_t u64Param2, size_t cb);
180;
181; @returns EFLAGS after the operation, only arithmetic flags are valid.
182; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
183; @param [esp + 08h] Param 2 - Second parameter.
184; @param [esp + 10h] Param 3 - Size of parameters, only 1/2/4/8 (64 bits host only) is valid.
185; @uses eax, ecx, edx
186;
187align 16
188BEGINPROC EMEmulateOr
189%ifdef RT_ARCH_AMD64
190%ifdef RT_OS_WINDOWS
191 mov rax, r8 ; eax = size of parameters
192%else ; !RT_OS_WINDOWS
193 mov rax, rdx ; rax = size of parameters
194 mov rcx, rdi ; rcx = first parameter
195 mov rdx, rsi ; rdx = second parameter
196%endif ; !RT_OS_WINDOWS
197%else ; !RT_ARCH_AMD64
198 mov eax, [esp + 10h] ; eax = size of parameters
199 mov ecx, [esp + 04h] ; ecx = first parameter
200 mov edx, [esp + 08h] ; edx = second parameter
201%endif
202
203 ; switch on size
204%ifdef RT_ARCH_AMD64
205 cmp al, 8
206 je short .do_qword ; 8 bytes variant
207%endif
208 cmp al, 4
209 je short .do_dword ; 4 bytes variant
210 cmp al, 2
211 je short .do_word ; 2 byte variant
212 cmp al, 1
213 je short .do_byte ; 1 bytes variant
214 int3
215
216 ; workers
217%ifdef RT_ARCH_AMD64
218.do_qword:
219 or [MY_PTR_REG], rdx ; do 8 bytes OR
220 jmp short .done
221%endif
222
223.do_dword:
224 or [MY_PTR_REG], edx ; do 4 bytes OR
225 jmp short .done
226
227.do_word:
228 or [MY_PTR_REG], dx ; do 2 bytes OR
229 jmp short .done
230
231.do_byte:
232 or [MY_PTR_REG], dl ; do 1 byte OR
233
234 ; collect flags and return.
235.done:
236 pushf
237 pop MY_RET_REG
238 retn
239ENDPROC EMEmulateOr
240
241;;
242; Emulate LOCK OR instruction.
243; EMDECL(int) EMEmulateLockOr(void *pvParam1, uint64_t u64Param2, size_t cbSize, RTGCUINTREG32 *pf);
244;
245; @returns VINF_SUCCESS on success, VERR_ACCESS_DENIED on \#PF (GC only).
246; @param [esp + 04h] gcc:rdi msc:rcx Param 1 - First parameter - pointer to data item (the real stuff).
247; @param [esp + 08h] gcc:rsi msc:rdx Param 2 - Second parameter- the immediate / register value.
248; @param [esp + 10h] gcc:rdx msc:r8 Param 3 - Size of the operation - 1, 2, 4 or 8 bytes.
249; @param [esp + 14h] gcc:rcx msc:r9 Param 4 - Where to store the eflags on success.
250; only arithmetic flags are valid.
251align 16
252BEGINPROC EMEmulateLockOr
253%ifdef RT_ARCH_AMD64
254%ifdef RT_OS_WINDOWS
255 mov rax, r8 ; eax = size of parameters
256%else ; !RT_OS_WINDOWS
257 mov rax, rdx ; rax = size of parameters
258 mov rcx, rdi ; rcx = first parameter
259 mov rdx, rsi ; rdx = second parameter
260%endif ; !RT_OS_WINDOWS
261%else ; !RT_ARCH_AMD64
262 mov eax, [esp + 10h] ; eax = size of parameters
263 mov ecx, [esp + 04h] ; ecx = first parameter (MY_PTR_REG)
264 mov edx, [esp + 08h] ; edx = second parameter
265%endif
266
267 ; switch on size
268%ifdef RT_ARCH_AMD64
269 cmp al, 8
270 je short .do_qword ; 8 bytes variant
271%endif
272 cmp al, 4
273 je short .do_dword ; 4 bytes variant
274 cmp al, 2
275 je short .do_word ; 2 byte variant
276 cmp al, 1
277 je short .do_byte ; 1 bytes variant
278 int3
279
280 ; workers
281%ifdef RT_ARCH_AMD64
282.do_qword:
283 lock or [MY_PTR_REG], rdx ; do 8 bytes OR
284 jmp short .done
285%endif
286
287.do_dword:
288 lock or [MY_PTR_REG], edx ; do 4 bytes OR
289 jmp short .done
290
291.do_word:
292 lock or [MY_PTR_REG], dx ; do 2 bytes OR
293 jmp short .done
294
295.do_byte:
296 lock or [MY_PTR_REG], dl ; do 1 byte OR
297
298 ; collect flags and return.
299.done:
300 pushf
301%ifdef RT_ARCH_AMD64
302 pop rax
303 %ifdef RT_OS_WINDOWS
304 mov [r9], eax
305 %else ; !RT_OS_WINDOWS
306 mov [rcx], eax
307 %endif ; !RT_OS_WINDOWS
308%else ; !RT_ARCH_AMD64
309 mov eax, [esp + 14h + 4]
310 pop dword [eax]
311%endif
312 mov eax, VINF_SUCCESS
313 retn
314
315%ifdef IN_GC
316; #PF resume point.
317GLOBALNAME EMEmulateLockOr_Error
318 mov eax, VERR_ACCESS_DENIED
319 ret
320%endif
321
322ENDPROC EMEmulateLockOr
323
324;;
325; Emulate XOR instruction, CDECL calling conv.
326; EMDECL(uint32_t) EMEmulateXor(void *pvParam1, uint64_t u64Param2, size_t cb);
327;
328; @returns EFLAGS after the operation, only arithmetic flags are valid.
329; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
330; @param [esp + 08h] Param 2 - Second parameter.
331; @param [esp + 10h] Param 3 - Size of parameters, only 1/2/4/8 (64 bits host only) is valid.
332; @uses eax, ecx, edx
333;
334align 16
335BEGINPROC EMEmulateXor
336%ifdef RT_ARCH_AMD64
337%ifdef RT_OS_WINDOWS
338 mov rax, r8 ; eax = size of parameters
339%else ; !RT_OS_WINDOWS
340 mov rax, rdx ; rax = size of parameters
341 mov rcx, rdi ; rcx = first parameter
342 mov rdx, rsi ; rdx = second parameter
343%endif ; !RT_OS_WINDOWS
344%else ; !RT_ARCH_AMD64
345 mov eax, [esp + 10h] ; eax = size of parameters
346 mov ecx, [esp + 04h] ; ecx = first parameter
347 mov edx, [esp + 08h] ; edx = second parameter
348%endif
349
350 ; switch on size
351%ifdef RT_ARCH_AMD64
352 cmp al, 8
353 je short .do_qword ; 8 bytes variant
354%endif
355 cmp al, 4
356 je short .do_dword ; 4 bytes variant
357 cmp al, 2
358 je short .do_word ; 2 byte variant
359 cmp al, 1
360 je short .do_byte ; 1 bytes variant
361 int3
362
363 ; workers
364%ifdef RT_ARCH_AMD64
365.do_qword:
366 xor [MY_PTR_REG], rdx ; do 8 bytes XOR
367 jmp short .done
368%endif
369
370.do_dword:
371 xor [MY_PTR_REG], edx ; do 4 bytes XOR
372 jmp short .done
373
374.do_word:
375 xor [MY_PTR_REG], dx ; do 2 bytes XOR
376 jmp short .done
377
378.do_byte:
379 xor [MY_PTR_REG], dl ; do 1 byte XOR
380
381 ; collect flags and return.
382.done:
383 pushf
384 pop MY_RET_REG
385 retn
386ENDPROC EMEmulateXor
387
388;;
389; Emulate INC instruction, CDECL calling conv.
390; EMDECL(uint32_t) EMEmulateInc(void *pvParam1, size_t cb);
391;
392; @returns EFLAGS after the operation, only arithmetic flags are valid.
393; @param [esp + 04h] rdi rcx Param 1 - First parameter - pointer to data item.
394; @param [esp + 08h] rsi rdx Param 2 - Size of parameters, only 1/2/4 is valid.
395; @uses eax, ecx, edx
396;
397align 16
398BEGINPROC EMEmulateInc
399%ifdef RT_ARCH_AMD64
400%ifdef RT_OS_WINDOWS
401 mov rax, rdx ; eax = size of parameters
402%else ; !RT_OS_WINDOWS
403 mov rax, rsi ; eax = size of parameters
404 mov rcx, rdi ; rcx = first parameter
405%endif ; !RT_OS_WINDOWS
406%else ; !RT_ARCH_AMD64
407 mov eax, [esp + 08h] ; eax = size of parameters
408 mov ecx, [esp + 04h] ; ecx = first parameter
409%endif
410
411 ; switch on size
412%ifdef RT_ARCH_AMD64
413 cmp al, 8
414 je short .do_qword ; 8 bytes variant
415%endif
416 cmp al, 4
417 je short .do_dword ; 4 bytes variant
418 cmp al, 2
419 je short .do_word ; 2 byte variant
420 cmp al, 1
421 je short .do_byte ; 1 bytes variant
422 int3
423
424 ; workers
425%ifdef RT_ARCH_AMD64
426.do_qword:
427 inc qword [MY_PTR_REG] ; do 8 bytes INC
428 jmp short .done
429%endif
430
431.do_dword:
432 inc dword [MY_PTR_REG] ; do 4 bytes INC
433 jmp short .done
434
435.do_word:
436 inc word [MY_PTR_REG] ; do 2 bytes INC
437 jmp short .done
438
439.do_byte:
440 inc byte [MY_PTR_REG] ; do 1 byte INC
441 jmp short .done
442
443 ; collect flags and return.
444.done:
445 pushf
446 pop MY_RET_REG
447 retn
448ENDPROC EMEmulateInc
449
450
451;;
452; Emulate DEC instruction, CDECL calling conv.
453; EMDECL(uint32_t) EMEmulateDec(void *pvParam1, size_t cb);
454;
455; @returns EFLAGS after the operation, only arithmetic flags are valid.
456; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
457; @param [esp + 08h] Param 2 - Size of parameters, only 1/2/4 is valid.
458; @uses eax, ecx, edx
459;
460align 16
461BEGINPROC EMEmulateDec
462%ifdef RT_ARCH_AMD64
463%ifdef RT_OS_WINDOWS
464 mov rax, rdx ; eax = size of parameters
465%else ; !RT_OS_WINDOWS
466 mov rax, rsi ; eax = size of parameters
467 mov rcx, rdi ; rcx = first parameter
468%endif ; !RT_OS_WINDOWS
469%else ; !RT_ARCH_AMD64
470 mov eax, [esp + 08h] ; eax = size of parameters
471 mov ecx, [esp + 04h] ; ecx = first parameter
472%endif
473
474 ; switch on size
475%ifdef RT_ARCH_AMD64
476 cmp al, 8
477 je short .do_qword ; 8 bytes variant
478%endif
479 cmp al, 4
480 je short .do_dword ; 4 bytes variant
481 cmp al, 2
482 je short .do_word ; 2 byte variant
483 cmp al, 1
484 je short .do_byte ; 1 bytes variant
485 int3
486
487 ; workers
488%ifdef RT_ARCH_AMD64
489.do_qword:
490 dec qword [MY_PTR_REG] ; do 8 bytes DEC
491 jmp short .done
492%endif
493
494.do_dword:
495 dec dword [MY_PTR_REG] ; do 4 bytes DEC
496 jmp short .done
497
498.do_word:
499 dec word [MY_PTR_REG] ; do 2 bytes DEC
500 jmp short .done
501
502.do_byte:
503 dec byte [MY_PTR_REG] ; do 1 byte DEC
504 jmp short .done
505
506 ; collect flags and return.
507.done:
508 pushf
509 pop MY_RET_REG
510 retn
511ENDPROC EMEmulateDec
512
513;;
514; Emulate ADD instruction, CDECL calling conv.
515; EMDECL(uint32_t) EMEmulateAdd(void *pvParam1, uint64_t u64Param2, size_t cb);
516;
517; @returns EFLAGS after the operation, only arithmetic flags are valid.
518; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
519; @param [esp + 08h] Param 2 - Second parameter.
520; @param [esp + 10h] Param 3 - Size of parameters, only 1/2/4/8 (64 bits host only) is valid.
521; @uses eax, ecx, edx
522;
523align 16
524BEGINPROC EMEmulateAdd
525%ifdef RT_ARCH_AMD64
526%ifdef RT_OS_WINDOWS
527 mov rax, r8 ; eax = size of parameters
528%else ; !RT_OS_WINDOWS
529 mov rax, rdx ; rax = size of parameters
530 mov rcx, rdi ; rcx = first parameter
531 mov rdx, rsi ; rdx = second parameter
532%endif ; !RT_OS_WINDOWS
533%else ; !RT_ARCH_AMD64
534 mov eax, [esp + 10h] ; eax = size of parameters
535 mov ecx, [esp + 04h] ; ecx = first parameter
536 mov edx, [esp + 08h] ; edx = second parameter
537%endif
538
539 ; switch on size
540%ifdef RT_ARCH_AMD64
541 cmp al, 8
542 je short .do_qword ; 8 bytes variant
543%endif
544 cmp al, 4
545 je short .do_dword ; 4 bytes variant
546 cmp al, 2
547 je short .do_word ; 2 byte variant
548 cmp al, 1
549 je short .do_byte ; 1 bytes variant
550 int3
551
552 ; workers
553%ifdef RT_ARCH_AMD64
554.do_qword:
555 add [MY_PTR_REG], rdx ; do 8 bytes ADD
556 jmp short .done
557%endif
558
559.do_dword:
560 add [MY_PTR_REG], edx ; do 4 bytes ADD
561 jmp short .done
562
563.do_word:
564 add [MY_PTR_REG], dx ; do 2 bytes ADD
565 jmp short .done
566
567.do_byte:
568 add [MY_PTR_REG], dl ; do 1 byte ADD
569
570 ; collect flags and return.
571.done:
572 pushf
573 pop MY_RET_REG
574 retn
575ENDPROC EMEmulateAdd
576
577;;
578; Emulate ADC instruction, CDECL calling conv.
579; EMDECL(uint32_t) EMEmulateAdcWithCarrySet(void *pvParam1, uint64_t u64Param2, size_t cb);
580;
581; @returns EFLAGS after the operation, only arithmetic flags are valid.
582; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
583; @param [esp + 08h] Param 2 - Second parameter.
584; @param [esp + 10h] Param 3 - Size of parameters, only 1/2/4/8 (64 bits host only) is valid.
585; @uses eax, ecx, edx
586;
587align 16
588BEGINPROC EMEmulateAdcWithCarrySet
589%ifdef RT_ARCH_AMD64
590%ifdef RT_OS_WINDOWS
591 mov rax, r8 ; eax = size of parameters
592%else ; !RT_OS_WINDOWS
593 mov rax, rdx ; rax = size of parameters
594 mov rcx, rdi ; rcx = first parameter
595 mov rdx, rsi ; rdx = second parameter
596%endif ; !RT_OS_WINDOWS
597%else ; !RT_ARCH_AMD64
598 mov eax, [esp + 10h] ; eax = size of parameters
599 mov ecx, [esp + 04h] ; ecx = first parameter
600 mov edx, [esp + 08h] ; edx = second parameter
601%endif
602
603 ; switch on size
604%ifdef RT_ARCH_AMD64
605 cmp al, 8
606 je short .do_qword ; 8 bytes variant
607%endif
608 cmp al, 4
609 je short .do_dword ; 4 bytes variant
610 cmp al, 2
611 je short .do_word ; 2 byte variant
612 cmp al, 1
613 je short .do_byte ; 1 bytes variant
614 int3
615
616 ; workers
617%ifdef RT_ARCH_AMD64
618.do_qword:
619 stc ; set carry flag
620 adc [MY_PTR_REG], rdx ; do 8 bytes ADC
621 jmp short .done
622%endif
623
624.do_dword:
625 stc ; set carry flag
626 adc [MY_PTR_REG], edx ; do 4 bytes ADC
627 jmp short .done
628
629.do_word:
630 stc ; set carry flag
631 adc [MY_PTR_REG], dx ; do 2 bytes ADC
632 jmp short .done
633
634.do_byte:
635 stc ; set carry flag
636 adc [MY_PTR_REG], dl ; do 1 byte ADC
637
638 ; collect flags and return.
639.done:
640 pushf
641 pop MY_RET_REG
642 retn
643ENDPROC EMEmulateAdcWithCarrySet
644
645;;
646; Emulate SUB instruction, CDECL calling conv.
647; EMDECL(uint32_t) EMEmulateSub(void *pvParam1, uint64_t u64Param2, size_t cb);
648;
649; @returns EFLAGS after the operation, only arithmetic flags are valid.
650; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
651; @param [esp + 08h] Param 2 - Second parameter.
652; @param [esp + 10h] Param 3 - Size of parameters, only 1/2/4/8 (64 bits host only) is valid.
653; @uses eax, ecx, edx
654;
655align 16
656BEGINPROC EMEmulateSub
657%ifdef RT_ARCH_AMD64
658%ifdef RT_OS_WINDOWS
659 mov rax, r8 ; eax = size of parameters
660%else ; !RT_OS_WINDOWS
661 mov rax, rdx ; rax = size of parameters
662 mov rcx, rdi ; rcx = first parameter
663 mov rdx, rsi ; rdx = second parameter
664%endif ; !RT_OS_WINDOWS
665%else ; !RT_ARCH_AMD64
666 mov eax, [esp + 10h] ; eax = size of parameters
667 mov ecx, [esp + 04h] ; ecx = first parameter
668 mov edx, [esp + 08h] ; edx = second parameter
669%endif
670
671 ; switch on size
672%ifdef RT_ARCH_AMD64
673 cmp al, 8
674 je short .do_qword ; 8 bytes variant
675%endif
676 cmp al, 4
677 je short .do_dword ; 4 bytes variant
678 cmp al, 2
679 je short .do_word ; 2 byte variant
680 cmp al, 1
681 je short .do_byte ; 1 bytes variant
682 int3
683
684 ; workers
685%ifdef RT_ARCH_AMD64
686.do_qword:
687 sub [MY_PTR_REG], rdx ; do 8 bytes SUB
688 jmp short .done
689%endif
690
691.do_dword:
692 sub [MY_PTR_REG], edx ; do 4 bytes SUB
693 jmp short .done
694
695.do_word:
696 sub [MY_PTR_REG], dx ; do 2 bytes SUB
697 jmp short .done
698
699.do_byte:
700 sub [MY_PTR_REG], dl ; do 1 byte SUB
701
702 ; collect flags and return.
703.done:
704 pushf
705 pop MY_RET_REG
706 retn
707ENDPROC EMEmulateSub
708
709
710;;
711; Emulate BTR instruction, CDECL calling conv.
712; EMDECL(uint32_t) EMEmulateBtr(void *pvParam1, uint64_t u64Param2);
713;
714; @returns EFLAGS after the operation, only arithmetic flags are valid.
715; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
716; @param [esp + 08h] Param 2 - Second parameter.
717; @uses eax, ecx, edx
718;
719align 16
720BEGINPROC EMEmulateBtr
721%ifdef RT_ARCH_AMD64
722%ifndef RT_OS_WINDOWS
723 mov rcx, rdi ; rcx = first parameter
724 mov rdx, rsi ; rdx = second parameter
725%endif ; !RT_OS_WINDOWS
726%else ; !RT_ARCH_AMD64
727 mov ecx, [esp + 04h] ; ecx = first parameter
728 mov edx, [esp + 08h] ; edx = second parameter
729%endif
730
731 and edx, 7
732 btr [MY_PTR_REG], edx
733
734 ; collect flags and return.
735 pushf
736 pop MY_RET_REG
737 retn
738ENDPROC EMEmulateBtr
739
740;;
741; Emulate LOCK BTR instruction.
742; EMDECL(int) EMEmulateLockBtr(void *pvParam1, uint64_t u64Param2, RTGCUINTREG32 *pf);
743;
744; @returns VINF_SUCCESS on success, VERR_ACCESS_DENIED on \#PF (GC only).
745; @param [esp + 04h] gcc:rdi msc:rcx Param 1 - First parameter - pointer to data item (the real stuff).
746; @param [esp + 08h] gcc:rsi msc:rdx Param 2 - Second parameter- the immediate / register value. (really an 8 byte value)
747; @param [esp + 10h] gcc:rdx msc:r8 Param 3 - Where to store the eflags on success.
748;
749align 16
750BEGINPROC EMEmulateLockBtr
751%ifdef RT_ARCH_AMD64
752 %ifdef RT_OS_WINDOWS
753 mov rax, r8 ; rax = third parameter
754 %else ; !RT_OS_WINDOWS
755 mov rcx, rdi ; rcx = first parameter
756 mov rax, rdx ; rax = third parameter
757 mov rdx, rsi ; rdx = second parameter
758 %endif ; !RT_OS_WINDOWS
759%else ; !RT_ARCH_AMD64
760 mov ecx, [esp + 04h] ; ecx = first parameter
761 mov edx, [esp + 08h] ; edx = second parameter
762 mov eax, [esp + 10h] ; eax = third parameter
763%endif
764
765 lock btr [MY_PTR_REG], edx
766
767 ; collect flags and return.
768 pushf
769 pop xDX
770 mov [xAX], edx
771 mov eax, VINF_SUCCESS
772 retn
773
774%ifdef IN_GC
775; #PF resume point.
776GLOBALNAME EMEmulateLockBtr_Error
777 mov eax, VERR_ACCESS_DENIED
778 ret
779%endif
780
781ENDPROC EMEmulateLockBtr
782
783;;
784; Emulate BTC instruction, CDECL calling conv.
785; EMDECL(uint32_t) EMEmulateBtc(void *pvParam1, uint64_t u64Param2);
786;
787; @returns EFLAGS after the operation, only arithmetic flags are valid.
788; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
789; @param [esp + 08h] Param 2 - Second parameter.
790; @uses eax, ecx, edx
791;
792align 16
793BEGINPROC EMEmulateBtc
794%ifdef RT_ARCH_AMD64
795%ifndef RT_OS_WINDOWS
796 mov rcx, rdi ; rcx = first parameter
797 mov rdx, rsi ; rdx = second parameter
798%endif ; !RT_OS_WINDOWS
799%else ; !RT_ARCH_AMD64
800 mov ecx, [esp + 04h] ; ecx = first parameter
801 mov edx, [esp + 08h] ; edx = second parameter
802%endif
803
804 and edx, 7
805 btc [MY_PTR_REG], edx
806
807 ; collect flags and return.
808 pushf
809 pop MY_RET_REG
810 retn
811ENDPROC EMEmulateBtc
812
813;;
814; Emulate BTS instruction, CDECL calling conv.
815; EMDECL(uint32_t) EMEmulateBts(void *pvParam1, uint64_t u64Param2);
816;
817; @returns EFLAGS after the operation, only arithmetic flags are valid.
818; @param [esp + 04h] Param 1 - First parameter - pointer to data item.
819; @param [esp + 08h] Param 2 - Second parameter.
820; @uses eax, ecx, edx
821;
822align 16
823BEGINPROC EMEmulateBts
824%ifdef RT_ARCH_AMD64
825%ifndef RT_OS_WINDOWS
826 mov rcx, rdi ; rcx = first parameter
827 mov rdx, rsi ; rdx = second parameter
828%endif ; !RT_OS_WINDOWS
829%else ; !RT_ARCH_AMD64
830 mov ecx, [esp + 04h] ; ecx = first parameter
831 mov edx, [esp + 08h] ; edx = second parameter
832%endif
833
834 and edx, 7
835 bts [MY_PTR_REG], edx
836
837 ; collect flags and return.
838 pushf
839 pop MY_RET_REG
840 retn
841ENDPROC EMEmulateBts
842
843
844;;
845; Emulate LOCK CMPXCHG instruction, CDECL calling conv.
846; EMDECL(uint32_t) EMEmulateLockCmpXchg(void *pvParam1, uint64_t *pu64Param2, uint64_t u64Param3, size_t cbSize);
847;
848; @returns EFLAGS after the operation, only arithmetic flags are valid.
849; @param [esp + 04h] gcc:rdi msc:rcx Param 1 - First parameter - pointer to first parameter
850; @param [esp + 08h] gcc:rsi msc:rdx Param 2 - pointer to second parameter (eax)
851; @param [esp + 0ch] gcc:rdx msc:r8 Param 3 - third parameter
852; @param [esp + 14h] gcc:rcx msc:r9 Param 4 - Size of parameters, only 1/2/4/8 is valid
853; @uses eax, ecx, edx
854;
855align 16
856BEGINPROC EMEmulateLockCmpXchg
857 push xBX
858%ifdef RT_ARCH_AMD64
859%ifdef RT_OS_WINDOWS
860 ; rcx contains the first parameter already
861 mov rbx, rdx ; rdx = 2nd parameter
862 mov rdx, r8 ; r8 = 3rd parameter
863 mov rax, r9 ; r9 = size of parameters
864%else
865 mov rax, rcx ; rcx = size of parameters (4th)
866 mov rcx, rdi ; rdi = 1st parameter
867 mov rbx, rsi ; rsi = second parameter
868 ;rdx contains the 3rd parameter already
869%endif ; !RT_OS_WINDOWS
870%else ; !RT_ARCH_AMD64
871 mov ecx, [esp + 04h + 4] ; ecx = first parameter
872 mov ebx, [esp + 08h + 4] ; ebx = 2nd parameter (eax)
873 mov edx, [esp + 0ch + 4] ; edx = third parameter
874 mov eax, [esp + 14h + 4] ; eax = size of parameters
875%endif
876
877%ifdef RT_ARCH_AMD64
878 cmp al, 8
879 je short .do_qword ; 8 bytes variant
880%endif
881 cmp al, 4
882 je short .do_dword ; 4 bytes variant
883 cmp al, 2
884 je short .do_word ; 2 byte variant
885 cmp al, 1
886 je short .do_byte ; 1 bytes variant
887 int3
888
889%ifdef RT_ARCH_AMD64
890.do_qword:
891 ; load 2nd parameter's value
892 mov rax, qword [rbx]
893
894 lock cmpxchg qword [rcx], rdx ; do 8 bytes CMPXCHG
895 mov qword [rbx], rax
896 jmp short .done
897%endif
898
899.do_dword:
900 ; load 2nd parameter's value
901 mov eax, dword [xBX]
902
903 lock cmpxchg dword [xCX], edx ; do 4 bytes CMPXCHG
904 mov dword [xBX], eax
905 jmp short .done
906
907.do_word:
908 ; load 2nd parameter's value
909 mov eax, dword [xBX]
910
911 lock cmpxchg word [xCX], dx ; do 2 bytes CMPXCHG
912 mov word [xBX], ax
913 jmp short .done
914
915.do_byte:
916 ; load 2nd parameter's value
917 mov eax, dword [xBX]
918
919 lock cmpxchg byte [xCX], dl ; do 1 byte CMPXCHG
920 mov byte [xBX], al
921
922.done:
923 ; collect flags and return.
924 pushf
925 pop MY_RET_REG
926
927 pop xBX
928 retn
929ENDPROC EMEmulateLockCmpXchg
930
931;;
932; Emulate CMPXCHG instruction, CDECL calling conv.
933; EMDECL(uint32_t) EMEmulateCmpXchg(void *pvParam1, uint64_t *pu32Param2, uint64_t u32Param3, size_t cbSize);
934;
935; @returns EFLAGS after the operation, only arithmetic flags are valid.
936; @param [esp + 04h] gcc:rdi msc:rcx Param 1 - First parameter - pointer to first parameter
937; @param [esp + 08h] gcc:rsi msc:rdx Param 2 - pointer to second parameter (eax)
938; @param [esp + 0ch] gcc:rdx msc:r8 Param 3 - third parameter
939; @param [esp + 14h] gcc:rcx msc:r9 Param 4 - Size of parameters, only 1/2/4 is valid.
940; @uses eax, ecx, edx
941;
942align 16
943BEGINPROC EMEmulateCmpXchg
944 push xBX
945%ifdef RT_ARCH_AMD64
946%ifdef RT_OS_WINDOWS
947 ; rcx contains the first parameter already
948 mov rbx, rdx ; rdx = 2nd parameter
949 mov rdx, r8 ; r8 = 3rd parameter
950 mov rax, r9 ; r9 = size of parameters
951%else
952 mov rax, rcx ; rcx = size of parameters (4th)
953 mov rcx, rdi ; rdi = 1st parameter
954 mov rbx, rsi ; rsi = second parameter
955 ;rdx contains the 3rd parameter already
956%endif ; !RT_OS_WINDOWS
957%else ; !RT_ARCH_AMD64
958 mov ecx, [esp + 04h + 4] ; ecx = first parameter
959 mov ebx, [esp + 08h + 4] ; ebx = 2nd parameter (eax)
960 mov edx, [esp + 0ch + 4] ; edx = third parameter
961 mov eax, [esp + 14h + 4] ; eax = size of parameters
962%endif
963
964%ifdef RT_ARCH_AMD64
965 cmp al, 8
966 je short .do_qword ; 8 bytes variant
967%endif
968 cmp al, 4
969 je short .do_dword ; 4 bytes variant
970 cmp al, 2
971 je short .do_word ; 2 byte variant
972 cmp al, 1
973 je short .do_byte ; 1 bytes variant
974 int3
975
976%ifdef RT_ARCH_AMD64
977.do_qword:
978 ; load 2nd parameter's value
979 mov rax, qword [rbx]
980
981 cmpxchg qword [rcx], rdx ; do 8 bytes CMPXCHG
982 mov qword [rbx], rax
983 jmp short .done
984%endif
985
986.do_dword:
987 ; load 2nd parameter's value
988 mov eax, dword [xBX]
989
990 cmpxchg dword [xCX], edx ; do 4 bytes CMPXCHG
991 mov dword [xBX], eax
992 jmp short .done
993
994.do_word:
995 ; load 2nd parameter's value
996 mov eax, dword [xBX]
997
998 cmpxchg word [xCX], dx ; do 2 bytes CMPXCHG
999 mov word [xBX], ax
1000 jmp short .done
1001
1002.do_byte:
1003 ; load 2nd parameter's value
1004 mov eax, dword [xBX]
1005
1006 cmpxchg byte [xCX], dl ; do 1 byte CMPXCHG
1007 mov byte [xBX], al
1008
1009.done:
1010 ; collect flags and return.
1011 pushf
1012 pop MY_RET_REG
1013
1014 pop xBX
1015 retn
1016ENDPROC EMEmulateCmpXchg
1017
1018%if 0
1019;; not tested!!
1020
1021;;
1022; Emulate LOCK CMPXCHG8B instruction, CDECL calling conv.
1023; EMDECL(uint32_t) EMEmulateLockCmpXchg8b(RTHCPTR pu32Param1, uint32_t *pEAX, uint32_t *pEDX, uint32_t uEBX, uint32_t uECX);
1024;
1025; @returns EFLAGS after the operation, only arithmetic flags are valid.
1026; @param [esp + 04h] Param 1 - First parameter - pointer to first parameter
1027; @param [esp + 08h] Param 2 - Address of the eax register
1028; @param [esp + 0ch] Param 3 - Address of the edx register
1029; @param [esp + 10h] Param 4 - EBX
1030; @param [esp + 14h] Param 5 - ECX
1031; @uses eax, ecx, edx
1032;
1033align 16
1034BEGINPROC EMEmulateLockCmpXchg8b
1035 push ebp
1036 push ebx
1037 mov ebp, [esp + 04h + 8] ; ebp = first parameter
1038 mov eax, [esp + 08h + 8] ; &EAX
1039 mov eax, dword [eax]
1040 mov edx, [esp + 0ch + 8] ; &EDX
1041 mov edx, dword [edx]
1042 mov ebx, [esp + 10h + 8] ; EBX
1043 mov ecx, [esp + 14h + 8] ; ECX
1044
1045 lock cmpxchg8b qword [ebp] ; do CMPXCHG8B
1046 mov ebx, dword [esp + 08h + 8]
1047 mov dword [ebx], eax
1048 mov ebx, dword [esp + 0ch + 8]
1049 mov dword [ebx], edx
1050
1051 ; collect flags and return.
1052 pushf
1053 pop eax
1054
1055 pop ebx
1056 pop ebp
1057 retn
1058
1059ENDPROC EMEmulateLockCmpXchg8b
1060
1061;;
1062; Emulate CMPXCHG8B instruction, CDECL calling conv.
1063; EMDECL(uint32_t) EMEmulateCmpXchg8b(RTHCPTR pu32Param1, uint32_t *pEAX, uint32_t *pEDX, uint32_t uEBX, uint32_t uECX);
1064;
1065; @returns EFLAGS after the operation, only arithmetic flags are valid.
1066; @param [esp + 04h] Param 1 - First parameter - pointer to first parameter
1067; @param [esp + 08h] Param 2 - Address of the eax register
1068; @param [esp + 0ch] Param 3 - Address of the edx register
1069; @param [esp + 10h] Param 4 - EBX
1070; @param [esp + 14h] Param 5 - ECX
1071; @uses eax, ecx, edx
1072;
1073align 16
1074BEGINPROC EMEmulateCmpXchg8b
1075 push ebp
1076 push ebx
1077 mov ebp, [esp + 04h + 8] ; ebp = first parameter
1078 mov eax, [esp + 08h + 8] ; &EAX
1079 mov eax, dword [eax]
1080 mov edx, [esp + 0ch + 8] ; &EDX
1081 mov edx, dword [edx]
1082 mov ebx, [esp + 10h + 8] ; EBX
1083 mov ecx, [esp + 14h + 8] ; ECX
1084
1085 cmpxchg8b qword [ebp] ; do CMPXCHG8B
1086 mov ebx, dword [esp + 08h + 8]
1087 mov dword [ebx], eax
1088 mov ebx, dword [esp + 0ch + 8]
1089 mov dword [ebx], edx
1090
1091 ; collect flags and return.
1092 pushf
1093 pop eax
1094
1095 pop ebx
1096 pop ebp
1097 retn
1098ENDPROC EMEmulateCmpXchg8b
1099
1100%endif
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