VirtualBox

source: vbox/trunk/include/iprt/asm-amd64-x86.h@ 48594

Last change on this file since 48594 was 48572, checked in by vboxsync, 11 years ago

asm-amd64-x86.h: Use writeeflags and readeflags with visual C++. Makes life simpler when building EFI.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 59.7 KB
Line 
1/** @file
2 * IPRT - AMD64 and x86 Specific Assembly Functions.
3 */
4
5/*
6 * Copyright (C) 2006-2013 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 * The contents of this file may alternatively be used under the terms
17 * of the Common Development and Distribution License Version 1.0
18 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
19 * VirtualBox OSE distribution, in which case the provisions of the
20 * CDDL are applicable instead of those of the GPL.
21 *
22 * You may elect to license modified versions of this file under the
23 * terms and conditions of either the GPL or the CDDL or both.
24 */
25
26#ifndef ___iprt_asm_amd64_x86_h
27#define ___iprt_asm_amd64_x86_h
28
29#include <iprt/types.h>
30#if !defined(RT_ARCH_AMD64) && !defined(RT_ARCH_X86)
31# error "Not on AMD64 or x86"
32#endif
33
34#if defined(_MSC_VER) && RT_INLINE_ASM_USES_INTRIN
35# include <intrin.h>
36 /* Emit the intrinsics at all optimization levels. */
37# pragma intrinsic(_ReadWriteBarrier)
38# pragma intrinsic(__cpuid)
39# pragma intrinsic(_enable)
40# pragma intrinsic(_disable)
41# pragma intrinsic(__rdtsc)
42# pragma intrinsic(__readmsr)
43# pragma intrinsic(__writemsr)
44# pragma intrinsic(__outbyte)
45# pragma intrinsic(__outbytestring)
46# pragma intrinsic(__outword)
47# pragma intrinsic(__outwordstring)
48# pragma intrinsic(__outdword)
49# pragma intrinsic(__outdwordstring)
50# pragma intrinsic(__inbyte)
51# pragma intrinsic(__inbytestring)
52# pragma intrinsic(__inword)
53# pragma intrinsic(__inwordstring)
54# pragma intrinsic(__indword)
55# pragma intrinsic(__indwordstring)
56# pragma intrinsic(__invlpg)
57# pragma intrinsic(__wbinvd)
58# pragma intrinsic(__readcr0)
59# pragma intrinsic(__readcr2)
60# pragma intrinsic(__readcr3)
61# pragma intrinsic(__readcr4)
62# pragma intrinsic(__writecr0)
63# pragma intrinsic(__writecr3)
64# pragma intrinsic(__writecr4)
65# pragma intrinsic(__readdr)
66# pragma intrinsic(__writedr)
67# ifdef RT_ARCH_AMD64
68# pragma intrinsic(__readcr8)
69# pragma intrinsic(__writecr8)
70# endif
71# if RT_INLINE_ASM_USES_INTRIN >= 15
72# pragma intrinsic(__readeflags)
73# pragma intrinsic(__writeeflags)
74# endif
75#endif
76
77
78
79/** @defgroup grp_rt_asm_amd64_x86 AMD64 and x86 Specific ASM Routines
80 * @ingroup grp_rt_asm
81 * @{
82 */
83
84/** @todo find a more proper place for this structure? */
85#pragma pack(1)
86/** IDTR */
87typedef struct RTIDTR
88{
89 /** Size of the IDT. */
90 uint16_t cbIdt;
91 /** Address of the IDT. */
92 uintptr_t pIdt;
93} RTIDTR, *PRTIDTR;
94#pragma pack()
95
96#pragma pack(1)
97/** GDTR */
98typedef struct RTGDTR
99{
100 /** Size of the GDT. */
101 uint16_t cbGdt;
102 /** Address of the GDT. */
103 uintptr_t pGdt;
104} RTGDTR, *PRTGDTR;
105#pragma pack()
106
107
108/**
109 * Gets the content of the IDTR CPU register.
110 * @param pIdtr Where to store the IDTR contents.
111 */
112#if RT_INLINE_ASM_EXTERNAL
113DECLASM(void) ASMGetIDTR(PRTIDTR pIdtr);
114#else
115DECLINLINE(void) ASMGetIDTR(PRTIDTR pIdtr)
116{
117# if RT_INLINE_ASM_GNU_STYLE
118 __asm__ __volatile__("sidt %0" : "=m" (*pIdtr));
119# else
120 __asm
121 {
122# ifdef RT_ARCH_AMD64
123 mov rax, [pIdtr]
124 sidt [rax]
125# else
126 mov eax, [pIdtr]
127 sidt [eax]
128# endif
129 }
130# endif
131}
132#endif
133
134
135/**
136 * Sets the content of the IDTR CPU register.
137 * @param pIdtr Where to load the IDTR contents from
138 */
139#if RT_INLINE_ASM_EXTERNAL
140DECLASM(void) ASMSetIDTR(const RTIDTR *pIdtr);
141#else
142DECLINLINE(void) ASMSetIDTR(const RTIDTR *pIdtr)
143{
144# if RT_INLINE_ASM_GNU_STYLE
145 __asm__ __volatile__("lidt %0" : : "m" (*pIdtr));
146# else
147 __asm
148 {
149# ifdef RT_ARCH_AMD64
150 mov rax, [pIdtr]
151 lidt [rax]
152# else
153 mov eax, [pIdtr]
154 lidt [eax]
155# endif
156 }
157# endif
158}
159#endif
160
161
162/**
163 * Gets the content of the GDTR CPU register.
164 * @param pGdtr Where to store the GDTR contents.
165 */
166#if RT_INLINE_ASM_EXTERNAL
167DECLASM(void) ASMGetGDTR(PRTGDTR pGdtr);
168#else
169DECLINLINE(void) ASMGetGDTR(PRTGDTR pGdtr)
170{
171# if RT_INLINE_ASM_GNU_STYLE
172 __asm__ __volatile__("sgdt %0" : "=m" (*pGdtr));
173# else
174 __asm
175 {
176# ifdef RT_ARCH_AMD64
177 mov rax, [pGdtr]
178 sgdt [rax]
179# else
180 mov eax, [pGdtr]
181 sgdt [eax]
182# endif
183 }
184# endif
185}
186#endif
187
188/**
189 * Get the cs register.
190 * @returns cs.
191 */
192#if RT_INLINE_ASM_EXTERNAL
193DECLASM(RTSEL) ASMGetCS(void);
194#else
195DECLINLINE(RTSEL) ASMGetCS(void)
196{
197 RTSEL SelCS;
198# if RT_INLINE_ASM_GNU_STYLE
199 __asm__ __volatile__("movw %%cs, %0\n\t" : "=r" (SelCS));
200# else
201 __asm
202 {
203 mov ax, cs
204 mov [SelCS], ax
205 }
206# endif
207 return SelCS;
208}
209#endif
210
211
212/**
213 * Get the DS register.
214 * @returns DS.
215 */
216#if RT_INLINE_ASM_EXTERNAL
217DECLASM(RTSEL) ASMGetDS(void);
218#else
219DECLINLINE(RTSEL) ASMGetDS(void)
220{
221 RTSEL SelDS;
222# if RT_INLINE_ASM_GNU_STYLE
223 __asm__ __volatile__("movw %%ds, %0\n\t" : "=r" (SelDS));
224# else
225 __asm
226 {
227 mov ax, ds
228 mov [SelDS], ax
229 }
230# endif
231 return SelDS;
232}
233#endif
234
235
236/**
237 * Get the ES register.
238 * @returns ES.
239 */
240#if RT_INLINE_ASM_EXTERNAL
241DECLASM(RTSEL) ASMGetES(void);
242#else
243DECLINLINE(RTSEL) ASMGetES(void)
244{
245 RTSEL SelES;
246# if RT_INLINE_ASM_GNU_STYLE
247 __asm__ __volatile__("movw %%es, %0\n\t" : "=r" (SelES));
248# else
249 __asm
250 {
251 mov ax, es
252 mov [SelES], ax
253 }
254# endif
255 return SelES;
256}
257#endif
258
259
260/**
261 * Get the FS register.
262 * @returns FS.
263 */
264#if RT_INLINE_ASM_EXTERNAL
265DECLASM(RTSEL) ASMGetFS(void);
266#else
267DECLINLINE(RTSEL) ASMGetFS(void)
268{
269 RTSEL SelFS;
270# if RT_INLINE_ASM_GNU_STYLE
271 __asm__ __volatile__("movw %%fs, %0\n\t" : "=r" (SelFS));
272# else
273 __asm
274 {
275 mov ax, fs
276 mov [SelFS], ax
277 }
278# endif
279 return SelFS;
280}
281# endif
282
283
284/**
285 * Get the GS register.
286 * @returns GS.
287 */
288#if RT_INLINE_ASM_EXTERNAL
289DECLASM(RTSEL) ASMGetGS(void);
290#else
291DECLINLINE(RTSEL) ASMGetGS(void)
292{
293 RTSEL SelGS;
294# if RT_INLINE_ASM_GNU_STYLE
295 __asm__ __volatile__("movw %%gs, %0\n\t" : "=r" (SelGS));
296# else
297 __asm
298 {
299 mov ax, gs
300 mov [SelGS], ax
301 }
302# endif
303 return SelGS;
304}
305#endif
306
307
308/**
309 * Get the SS register.
310 * @returns SS.
311 */
312#if RT_INLINE_ASM_EXTERNAL
313DECLASM(RTSEL) ASMGetSS(void);
314#else
315DECLINLINE(RTSEL) ASMGetSS(void)
316{
317 RTSEL SelSS;
318# if RT_INLINE_ASM_GNU_STYLE
319 __asm__ __volatile__("movw %%ss, %0\n\t" : "=r" (SelSS));
320# else
321 __asm
322 {
323 mov ax, ss
324 mov [SelSS], ax
325 }
326# endif
327 return SelSS;
328}
329#endif
330
331
332/**
333 * Get the TR register.
334 * @returns TR.
335 */
336#if RT_INLINE_ASM_EXTERNAL
337DECLASM(RTSEL) ASMGetTR(void);
338#else
339DECLINLINE(RTSEL) ASMGetTR(void)
340{
341 RTSEL SelTR;
342# if RT_INLINE_ASM_GNU_STYLE
343 __asm__ __volatile__("str %w0\n\t" : "=r" (SelTR));
344# else
345 __asm
346 {
347 str ax
348 mov [SelTR], ax
349 }
350# endif
351 return SelTR;
352}
353#endif
354
355
356/**
357 * Get the LDTR register.
358 * @returns LDTR.
359 */
360#if RT_INLINE_ASM_EXTERNAL
361DECLASM(RTSEL) ASMGetLDTR(void);
362#else
363DECLINLINE(RTSEL) ASMGetLDTR(void)
364{
365 RTSEL SelLDTR;
366# if RT_INLINE_ASM_GNU_STYLE
367 __asm__ __volatile__("sldt %w0\n\t" : "=r" (SelLDTR));
368# else
369 __asm
370 {
371 sldt ax
372 mov [SelLDTR], ax
373 }
374# endif
375 return SelLDTR;
376}
377#endif
378
379
380/**
381 * Get the [RE]FLAGS register.
382 * @returns [RE]FLAGS.
383 */
384#if RT_INLINE_ASM_EXTERNAL && RT_INLINE_ASM_USES_INTRIN < 15
385DECLASM(RTCCUINTREG) ASMGetFlags(void);
386#else
387DECLINLINE(RTCCUINTREG) ASMGetFlags(void)
388{
389 RTCCUINTREG uFlags;
390# if RT_INLINE_ASM_GNU_STYLE
391# ifdef RT_ARCH_AMD64
392 __asm__ __volatile__("pushfq\n\t"
393 "popq %0\n\t"
394 : "=r" (uFlags));
395# else
396 __asm__ __volatile__("pushfl\n\t"
397 "popl %0\n\t"
398 : "=r" (uFlags));
399# endif
400# elif RT_INLINE_ASM_USES_INTRIN >= 15
401 uFlags = __readeflags();
402# else
403 __asm
404 {
405# ifdef RT_ARCH_AMD64
406 pushfq
407 pop [uFlags]
408# else
409 pushfd
410 pop [uFlags]
411# endif
412 }
413# endif
414 return uFlags;
415}
416#endif
417
418
419/**
420 * Set the [RE]FLAGS register.
421 * @param uFlags The new [RE]FLAGS value.
422 */
423#if RT_INLINE_ASM_EXTERNAL && RT_INLINE_ASM_USES_INTRIN < 15
424DECLASM(void) ASMSetFlags(RTCCUINTREG uFlags);
425#else
426DECLINLINE(void) ASMSetFlags(RTCCUINTREG uFlags)
427{
428# if RT_INLINE_ASM_GNU_STYLE
429# ifdef RT_ARCH_AMD64
430 __asm__ __volatile__("pushq %0\n\t"
431 "popfq\n\t"
432 : : "g" (uFlags));
433# else
434 __asm__ __volatile__("pushl %0\n\t"
435 "popfl\n\t"
436 : : "g" (uFlags));
437# endif
438# elif RT_INLINE_ASM_USES_INTRIN >= 15
439 __writeeflags(uFlags);
440# else
441 __asm
442 {
443# ifdef RT_ARCH_AMD64
444 push [uFlags]
445 popfq
446# else
447 push [uFlags]
448 popfd
449# endif
450 }
451# endif
452}
453#endif
454
455
456/**
457 * Gets the content of the CPU timestamp counter register.
458 *
459 * @returns TSC.
460 */
461#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
462DECLASM(uint64_t) ASMReadTSC(void);
463#else
464DECLINLINE(uint64_t) ASMReadTSC(void)
465{
466 RTUINT64U u;
467# if RT_INLINE_ASM_GNU_STYLE
468 __asm__ __volatile__("rdtsc\n\t" : "=a" (u.s.Lo), "=d" (u.s.Hi));
469# else
470# if RT_INLINE_ASM_USES_INTRIN
471 u.u = __rdtsc();
472# else
473 __asm
474 {
475 rdtsc
476 mov [u.s.Lo], eax
477 mov [u.s.Hi], edx
478 }
479# endif
480# endif
481 return u.u;
482}
483#endif
484
485
486/**
487 * Performs the cpuid instruction returning all registers.
488 *
489 * @param uOperator CPUID operation (eax).
490 * @param pvEAX Where to store eax.
491 * @param pvEBX Where to store ebx.
492 * @param pvECX Where to store ecx.
493 * @param pvEDX Where to store edx.
494 * @remark We're using void pointers to ease the use of special bitfield structures and such.
495 */
496#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
497DECLASM(void) ASMCpuId(uint32_t uOperator, void *pvEAX, void *pvEBX, void *pvECX, void *pvEDX);
498#else
499DECLINLINE(void) ASMCpuId(uint32_t uOperator, void *pvEAX, void *pvEBX, void *pvECX, void *pvEDX)
500{
501# if RT_INLINE_ASM_GNU_STYLE
502# ifdef RT_ARCH_AMD64
503 RTCCUINTREG uRAX, uRBX, uRCX, uRDX;
504 __asm__ __volatile__ ("cpuid\n\t"
505 : "=a" (uRAX),
506 "=b" (uRBX),
507 "=c" (uRCX),
508 "=d" (uRDX)
509 : "0" (uOperator), "2" (0));
510 *(uint32_t *)pvEAX = (uint32_t)uRAX;
511 *(uint32_t *)pvEBX = (uint32_t)uRBX;
512 *(uint32_t *)pvECX = (uint32_t)uRCX;
513 *(uint32_t *)pvEDX = (uint32_t)uRDX;
514# else
515 __asm__ __volatile__ ("xchgl %%ebx, %1\n\t"
516 "cpuid\n\t"
517 "xchgl %%ebx, %1\n\t"
518 : "=a" (*(uint32_t *)pvEAX),
519 "=r" (*(uint32_t *)pvEBX),
520 "=c" (*(uint32_t *)pvECX),
521 "=d" (*(uint32_t *)pvEDX)
522 : "0" (uOperator), "2" (0));
523# endif
524
525# elif RT_INLINE_ASM_USES_INTRIN
526 int aInfo[4];
527 __cpuid(aInfo, uOperator);
528 *(uint32_t *)pvEAX = aInfo[0];
529 *(uint32_t *)pvEBX = aInfo[1];
530 *(uint32_t *)pvECX = aInfo[2];
531 *(uint32_t *)pvEDX = aInfo[3];
532
533# else
534 uint32_t uEAX;
535 uint32_t uEBX;
536 uint32_t uECX;
537 uint32_t uEDX;
538 __asm
539 {
540 push ebx
541 mov eax, [uOperator]
542 cpuid
543 mov [uEAX], eax
544 mov [uEBX], ebx
545 mov [uECX], ecx
546 mov [uEDX], edx
547 pop ebx
548 }
549 *(uint32_t *)pvEAX = uEAX;
550 *(uint32_t *)pvEBX = uEBX;
551 *(uint32_t *)pvECX = uECX;
552 *(uint32_t *)pvEDX = uEDX;
553# endif
554}
555#endif
556
557
558/**
559 * Performs the CPUID instruction with EAX and ECX input returning ALL output
560 * registers.
561 *
562 * @param uOperator CPUID operation (eax).
563 * @param uIdxECX ecx index
564 * @param pvEAX Where to store eax.
565 * @param pvEBX Where to store ebx.
566 * @param pvECX Where to store ecx.
567 * @param pvEDX Where to store edx.
568 * @remark We're using void pointers to ease the use of special bitfield structures and such.
569 */
570#if RT_INLINE_ASM_EXTERNAL || RT_INLINE_ASM_USES_INTRIN
571DECLASM(void) ASMCpuId_Idx_ECX(uint32_t uOperator, uint32_t uIdxECX, void *pvEAX, void *pvEBX, void *pvECX, void *pvEDX);
572#else
573DECLINLINE(void) ASMCpuId_Idx_ECX(uint32_t uOperator, uint32_t uIdxECX, void *pvEAX, void *pvEBX, void *pvECX, void *pvEDX)
574{
575# if RT_INLINE_ASM_GNU_STYLE
576# ifdef RT_ARCH_AMD64
577 RTCCUINTREG uRAX, uRBX, uRCX, uRDX;
578 __asm__ ("cpuid\n\t"
579 : "=a" (uRAX),
580 "=b" (uRBX),
581 "=c" (uRCX),
582 "=d" (uRDX)
583 : "0" (uOperator),
584 "2" (uIdxECX));
585 *(uint32_t *)pvEAX = (uint32_t)uRAX;
586 *(uint32_t *)pvEBX = (uint32_t)uRBX;
587 *(uint32_t *)pvECX = (uint32_t)uRCX;
588 *(uint32_t *)pvEDX = (uint32_t)uRDX;
589# else
590 __asm__ ("xchgl %%ebx, %1\n\t"
591 "cpuid\n\t"
592 "xchgl %%ebx, %1\n\t"
593 : "=a" (*(uint32_t *)pvEAX),
594 "=r" (*(uint32_t *)pvEBX),
595 "=c" (*(uint32_t *)pvECX),
596 "=d" (*(uint32_t *)pvEDX)
597 : "0" (uOperator),
598 "2" (uIdxECX));
599# endif
600
601# elif RT_INLINE_ASM_USES_INTRIN
602 int aInfo[4];
603 __cpuidex(aInfo, uOperator, uIdxECX);
604 *(uint32_t *)pvEAX = aInfo[0];
605 *(uint32_t *)pvEBX = aInfo[1];
606 *(uint32_t *)pvECX = aInfo[2];
607 *(uint32_t *)pvEDX = aInfo[3];
608
609# else
610 uint32_t uEAX;
611 uint32_t uEBX;
612 uint32_t uECX;
613 uint32_t uEDX;
614 __asm
615 {
616 push ebx
617 mov eax, [uOperator]
618 mov ecx, [uIdxECX]
619 cpuid
620 mov [uEAX], eax
621 mov [uEBX], ebx
622 mov [uECX], ecx
623 mov [uEDX], edx
624 pop ebx
625 }
626 *(uint32_t *)pvEAX = uEAX;
627 *(uint32_t *)pvEBX = uEBX;
628 *(uint32_t *)pvECX = uECX;
629 *(uint32_t *)pvEDX = uEDX;
630# endif
631}
632#endif
633
634
635/**
636 * Performs the cpuid instruction returning ecx and edx.
637 *
638 * @param uOperator CPUID operation (eax).
639 * @param pvECX Where to store ecx.
640 * @param pvEDX Where to store edx.
641 * @remark We're using void pointers to ease the use of special bitfield structures and such.
642 */
643#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
644DECLASM(void) ASMCpuId_ECX_EDX(uint32_t uOperator, void *pvECX, void *pvEDX);
645#else
646DECLINLINE(void) ASMCpuId_ECX_EDX(uint32_t uOperator, void *pvECX, void *pvEDX)
647{
648 uint32_t uEBX;
649 ASMCpuId(uOperator, &uOperator, &uEBX, pvECX, pvEDX);
650}
651#endif
652
653
654/**
655 * Performs the cpuid instruction returning eax.
656 *
657 * @param uOperator CPUID operation (eax).
658 * @returns EAX after cpuid operation.
659 */
660#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
661DECLASM(uint32_t) ASMCpuId_EAX(uint32_t uOperator);
662#else
663DECLINLINE(uint32_t) ASMCpuId_EAX(uint32_t uOperator)
664{
665 RTCCUINTREG xAX;
666# if RT_INLINE_ASM_GNU_STYLE
667# ifdef RT_ARCH_AMD64
668 __asm__ ("cpuid"
669 : "=a" (xAX)
670 : "0" (uOperator)
671 : "rbx", "rcx", "rdx");
672# elif (defined(PIC) || defined(__PIC__)) && defined(__i386__)
673 __asm__ ("push %%ebx\n\t"
674 "cpuid\n\t"
675 "pop %%ebx\n\t"
676 : "=a" (xAX)
677 : "0" (uOperator)
678 : "ecx", "edx");
679# else
680 __asm__ ("cpuid"
681 : "=a" (xAX)
682 : "0" (uOperator)
683 : "edx", "ecx", "ebx");
684# endif
685
686# elif RT_INLINE_ASM_USES_INTRIN
687 int aInfo[4];
688 __cpuid(aInfo, uOperator);
689 xAX = aInfo[0];
690
691# else
692 __asm
693 {
694 push ebx
695 mov eax, [uOperator]
696 cpuid
697 mov [xAX], eax
698 pop ebx
699 }
700# endif
701 return (uint32_t)xAX;
702}
703#endif
704
705
706/**
707 * Performs the cpuid instruction returning ebx.
708 *
709 * @param uOperator CPUID operation (eax).
710 * @returns EBX after cpuid operation.
711 */
712#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
713DECLASM(uint32_t) ASMCpuId_EBX(uint32_t uOperator);
714#else
715DECLINLINE(uint32_t) ASMCpuId_EBX(uint32_t uOperator)
716{
717 RTCCUINTREG xBX;
718# if RT_INLINE_ASM_GNU_STYLE
719# ifdef RT_ARCH_AMD64
720 RTCCUINTREG uSpill;
721 __asm__ ("cpuid"
722 : "=a" (uSpill),
723 "=b" (xBX)
724 : "0" (uOperator)
725 : "rdx", "rcx");
726# elif (defined(PIC) || defined(__PIC__)) && defined(__i386__)
727 __asm__ ("push %%ebx\n\t"
728 "cpuid\n\t"
729 "mov %%ebx, %%edx\n\t"
730 "pop %%ebx\n\t"
731 : "=a" (uOperator),
732 "=d" (xBX)
733 : "0" (uOperator)
734 : "ecx");
735# else
736 __asm__ ("cpuid"
737 : "=a" (uOperator),
738 "=b" (xBX)
739 : "0" (uOperator)
740 : "edx", "ecx");
741# endif
742
743# elif RT_INLINE_ASM_USES_INTRIN
744 int aInfo[4];
745 __cpuid(aInfo, uOperator);
746 xBX = aInfo[1];
747
748# else
749 __asm
750 {
751 push ebx
752 mov eax, [uOperator]
753 cpuid
754 mov [xBX], ebx
755 pop ebx
756 }
757# endif
758 return (uint32_t)xBX;
759}
760#endif
761
762
763/**
764 * Performs the cpuid instruction returning ecx.
765 *
766 * @param uOperator CPUID operation (eax).
767 * @returns ECX after cpuid operation.
768 */
769#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
770DECLASM(uint32_t) ASMCpuId_ECX(uint32_t uOperator);
771#else
772DECLINLINE(uint32_t) ASMCpuId_ECX(uint32_t uOperator)
773{
774 RTCCUINTREG xCX;
775# if RT_INLINE_ASM_GNU_STYLE
776# ifdef RT_ARCH_AMD64
777 RTCCUINTREG uSpill;
778 __asm__ ("cpuid"
779 : "=a" (uSpill),
780 "=c" (xCX)
781 : "0" (uOperator)
782 : "rbx", "rdx");
783# elif (defined(PIC) || defined(__PIC__)) && defined(__i386__)
784 __asm__ ("push %%ebx\n\t"
785 "cpuid\n\t"
786 "pop %%ebx\n\t"
787 : "=a" (uOperator),
788 "=c" (xCX)
789 : "0" (uOperator)
790 : "edx");
791# else
792 __asm__ ("cpuid"
793 : "=a" (uOperator),
794 "=c" (xCX)
795 : "0" (uOperator)
796 : "ebx", "edx");
797
798# endif
799
800# elif RT_INLINE_ASM_USES_INTRIN
801 int aInfo[4];
802 __cpuid(aInfo, uOperator);
803 xCX = aInfo[2];
804
805# else
806 __asm
807 {
808 push ebx
809 mov eax, [uOperator]
810 cpuid
811 mov [xCX], ecx
812 pop ebx
813 }
814# endif
815 return (uint32_t)xCX;
816}
817#endif
818
819
820/**
821 * Performs the cpuid instruction returning edx.
822 *
823 * @param uOperator CPUID operation (eax).
824 * @returns EDX after cpuid operation.
825 */
826#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
827DECLASM(uint32_t) ASMCpuId_EDX(uint32_t uOperator);
828#else
829DECLINLINE(uint32_t) ASMCpuId_EDX(uint32_t uOperator)
830{
831 RTCCUINTREG xDX;
832# if RT_INLINE_ASM_GNU_STYLE
833# ifdef RT_ARCH_AMD64
834 RTCCUINTREG uSpill;
835 __asm__ ("cpuid"
836 : "=a" (uSpill),
837 "=d" (xDX)
838 : "0" (uOperator)
839 : "rbx", "rcx");
840# elif (defined(PIC) || defined(__PIC__)) && defined(__i386__)
841 __asm__ ("push %%ebx\n\t"
842 "cpuid\n\t"
843 "pop %%ebx\n\t"
844 : "=a" (uOperator),
845 "=d" (xDX)
846 : "0" (uOperator)
847 : "ecx");
848# else
849 __asm__ ("cpuid"
850 : "=a" (uOperator),
851 "=d" (xDX)
852 : "0" (uOperator)
853 : "ebx", "ecx");
854# endif
855
856# elif RT_INLINE_ASM_USES_INTRIN
857 int aInfo[4];
858 __cpuid(aInfo, uOperator);
859 xDX = aInfo[3];
860
861# else
862 __asm
863 {
864 push ebx
865 mov eax, [uOperator]
866 cpuid
867 mov [xDX], edx
868 pop ebx
869 }
870# endif
871 return (uint32_t)xDX;
872}
873#endif
874
875
876/**
877 * Checks if the current CPU supports CPUID.
878 *
879 * @returns true if CPUID is supported.
880 */
881DECLINLINE(bool) ASMHasCpuId(void)
882{
883#ifdef RT_ARCH_AMD64
884 return true; /* ASSUME that all amd64 compatible CPUs have cpuid. */
885#else /* !RT_ARCH_AMD64 */
886 bool fRet = false;
887# if RT_INLINE_ASM_GNU_STYLE
888 uint32_t u1;
889 uint32_t u2;
890 __asm__ ("pushf\n\t"
891 "pop %1\n\t"
892 "mov %1, %2\n\t"
893 "xorl $0x200000, %1\n\t"
894 "push %1\n\t"
895 "popf\n\t"
896 "pushf\n\t"
897 "pop %1\n\t"
898 "cmpl %1, %2\n\t"
899 "setne %0\n\t"
900 "push %2\n\t"
901 "popf\n\t"
902 : "=m" (fRet), "=r" (u1), "=r" (u2));
903# else
904 __asm
905 {
906 pushfd
907 pop eax
908 mov ebx, eax
909 xor eax, 0200000h
910 push eax
911 popfd
912 pushfd
913 pop eax
914 cmp eax, ebx
915 setne fRet
916 push ebx
917 popfd
918 }
919# endif
920 return fRet;
921#endif /* !RT_ARCH_AMD64 */
922}
923
924
925/**
926 * Gets the APIC ID of the current CPU.
927 *
928 * @returns the APIC ID.
929 */
930#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
931DECLASM(uint8_t) ASMGetApicId(void);
932#else
933DECLINLINE(uint8_t) ASMGetApicId(void)
934{
935 RTCCUINTREG xBX;
936# if RT_INLINE_ASM_GNU_STYLE
937# ifdef RT_ARCH_AMD64
938 RTCCUINTREG uSpill;
939 __asm__ __volatile__ ("cpuid"
940 : "=a" (uSpill),
941 "=b" (xBX)
942 : "0" (1)
943 : "rcx", "rdx");
944# elif (defined(PIC) || defined(__PIC__)) && defined(__i386__)
945 RTCCUINTREG uSpill;
946 __asm__ __volatile__ ("mov %%ebx,%1\n\t"
947 "cpuid\n\t"
948 "xchgl %%ebx,%1\n\t"
949 : "=a" (uSpill),
950 "=rm" (xBX)
951 : "0" (1)
952 : "ecx", "edx");
953# else
954 RTCCUINTREG uSpill;
955 __asm__ __volatile__ ("cpuid"
956 : "=a" (uSpill),
957 "=b" (xBX)
958 : "0" (1)
959 : "ecx", "edx");
960# endif
961
962# elif RT_INLINE_ASM_USES_INTRIN
963 int aInfo[4];
964 __cpuid(aInfo, 1);
965 xBX = aInfo[1];
966
967# else
968 __asm
969 {
970 push ebx
971 mov eax, 1
972 cpuid
973 mov [xBX], ebx
974 pop ebx
975 }
976# endif
977 return (uint8_t)(xBX >> 24);
978}
979#endif
980
981
982/**
983 * Tests if it a genuine Intel CPU based on the ASMCpuId(0) output.
984 *
985 * @returns true/false.
986 * @param uEBX EBX return from ASMCpuId(0)
987 * @param uECX ECX return from ASMCpuId(0)
988 * @param uEDX EDX return from ASMCpuId(0)
989 */
990DECLINLINE(bool) ASMIsIntelCpuEx(uint32_t uEBX, uint32_t uECX, uint32_t uEDX)
991{
992 return uEBX == UINT32_C(0x756e6547)
993 && uECX == UINT32_C(0x6c65746e)
994 && uEDX == UINT32_C(0x49656e69);
995}
996
997
998/**
999 * Tests if this is a genuine Intel CPU.
1000 *
1001 * @returns true/false.
1002 * @remarks ASSUMES that cpuid is supported by the CPU.
1003 */
1004DECLINLINE(bool) ASMIsIntelCpu(void)
1005{
1006 uint32_t uEAX, uEBX, uECX, uEDX;
1007 ASMCpuId(0, &uEAX, &uEBX, &uECX, &uEDX);
1008 return ASMIsIntelCpuEx(uEBX, uECX, uEDX);
1009}
1010
1011
1012/**
1013 * Tests if it an authentic AMD CPU based on the ASMCpuId(0) output.
1014 *
1015 * @returns true/false.
1016 * @param uEBX EBX return from ASMCpuId(0)
1017 * @param uECX ECX return from ASMCpuId(0)
1018 * @param uEDX EDX return from ASMCpuId(0)
1019 */
1020DECLINLINE(bool) ASMIsAmdCpuEx(uint32_t uEBX, uint32_t uECX, uint32_t uEDX)
1021{
1022 return uEBX == UINT32_C(0x68747541)
1023 && uECX == UINT32_C(0x444d4163)
1024 && uEDX == UINT32_C(0x69746e65);
1025}
1026
1027
1028/**
1029 * Tests if this is an authentic AMD CPU.
1030 *
1031 * @returns true/false.
1032 * @remarks ASSUMES that cpuid is supported by the CPU.
1033 */
1034DECLINLINE(bool) ASMIsAmdCpu(void)
1035{
1036 uint32_t uEAX, uEBX, uECX, uEDX;
1037 ASMCpuId(0, &uEAX, &uEBX, &uECX, &uEDX);
1038 return ASMIsAmdCpuEx(uEBX, uECX, uEDX);
1039}
1040
1041
1042/**
1043 * Tests if it a centaur hauling VIA CPU based on the ASMCpuId(0) output.
1044 *
1045 * @returns true/false.
1046 * @param uEBX EBX return from ASMCpuId(0).
1047 * @param uECX ECX return from ASMCpuId(0).
1048 * @param uEDX EDX return from ASMCpuId(0).
1049 */
1050DECLINLINE(bool) ASMIsViaCentaurCpuEx(uint32_t uEBX, uint32_t uECX, uint32_t uEDX)
1051{
1052 return uEBX == UINT32_C(0x746e6543)
1053 && uECX == UINT32_C(0x736c7561)
1054 && uEDX == UINT32_C(0x48727561);
1055}
1056
1057
1058/**
1059 * Tests if this is a centaur hauling VIA CPU.
1060 *
1061 * @returns true/false.
1062 * @remarks ASSUMES that cpuid is supported by the CPU.
1063 */
1064DECLINLINE(bool) ASMIsViaCentaurCpu(void)
1065{
1066 uint32_t uEAX, uEBX, uECX, uEDX;
1067 ASMCpuId(0, &uEAX, &uEBX, &uECX, &uEDX);
1068 return ASMIsAmdCpuEx(uEBX, uECX, uEDX);
1069}
1070
1071
1072/**
1073 * Checks whether ASMCpuId_EAX(0x00000000) indicates a valid range.
1074 *
1075 *
1076 * @returns true/false.
1077 * @param uEAX The EAX value of CPUID leaf 0x00000000.
1078 *
1079 * @note This only succeeds if there are at least two leaves in the range.
1080 * @remarks The upper range limit is just some half reasonable value we've
1081 * picked out of thin air.
1082 */
1083DECLINLINE(bool) ASMIsValidStdRange(uint32_t uEAX)
1084{
1085 return uEAX >= UINT32_C(0x00000001) && uEAX <= UINT32_C(0x000fffff);
1086}
1087
1088
1089/**
1090 * Checks whether ASMCpuId_EAX(0x80000000) indicates a valid range.
1091 *
1092 * This only succeeds if there are at least two leaves in the range.
1093 *
1094 * @returns true/false.
1095 * @param uEAX The EAX value of CPUID leaf 0x80000000.
1096 *
1097 * @note This only succeeds if there are at least two leaves in the range.
1098 * @remarks The upper range limit is just some half reasonable value we've
1099 * picked out of thin air.
1100 */
1101DECLINLINE(bool) ASMIsValidExtRange(uint32_t uEAX)
1102{
1103 return uEAX >= UINT32_C(0x80000001) && uEAX <= UINT32_C(0x800fffff);
1104}
1105
1106
1107/**
1108 * Extracts the CPU family from ASMCpuId(1) or ASMCpuId(0x80000001)
1109 *
1110 * @returns Family.
1111 * @param uEAX EAX return from ASMCpuId(1) or ASMCpuId(0x80000001).
1112 */
1113DECLINLINE(uint32_t) ASMGetCpuFamily(uint32_t uEAX)
1114{
1115 return ((uEAX >> 8) & 0xf) == 0xf
1116 ? ((uEAX >> 20) & 0x7f) + 0xf
1117 : ((uEAX >> 8) & 0xf);
1118}
1119
1120
1121/**
1122 * Extracts the CPU model from ASMCpuId(1) or ASMCpuId(0x80000001), Intel variant.
1123 *
1124 * @returns Model.
1125 * @param uEAX EAX from ASMCpuId(1) or ASMCpuId(0x80000001).
1126 */
1127DECLINLINE(uint32_t) ASMGetCpuModelIntel(uint32_t uEAX)
1128{
1129 return ((uEAX >> 8) & 0xf) == 0xf || (((uEAX >> 8) & 0xf) == 0x6) /* family! */
1130 ? ((uEAX >> 4) & 0xf) | ((uEAX >> 12) & 0xf0)
1131 : ((uEAX >> 4) & 0xf);
1132}
1133
1134
1135/**
1136 * Extracts the CPU model from ASMCpuId(1) or ASMCpuId(0x80000001), AMD variant.
1137 *
1138 * @returns Model.
1139 * @param uEAX EAX from ASMCpuId(1) or ASMCpuId(0x80000001).
1140 */
1141DECLINLINE(uint32_t) ASMGetCpuModelAMD(uint32_t uEAX)
1142{
1143 return ((uEAX >> 8) & 0xf) == 0xf
1144 ? ((uEAX >> 4) & 0xf) | ((uEAX >> 12) & 0xf0)
1145 : ((uEAX >> 4) & 0xf);
1146}
1147
1148
1149/**
1150 * Extracts the CPU model from ASMCpuId(1) or ASMCpuId(0x80000001)
1151 *
1152 * @returns Model.
1153 * @param uEAX EAX from ASMCpuId(1) or ASMCpuId(0x80000001).
1154 * @param fIntel Whether it's an intel CPU. Use ASMIsIntelCpuEx() or ASMIsIntelCpu().
1155 */
1156DECLINLINE(uint32_t) ASMGetCpuModel(uint32_t uEAX, bool fIntel)
1157{
1158 return ((uEAX >> 8) & 0xf) == 0xf || (((uEAX >> 8) & 0xf) == 0x6 && fIntel) /* family! */
1159 ? ((uEAX >> 4) & 0xf) | ((uEAX >> 12) & 0xf0)
1160 : ((uEAX >> 4) & 0xf);
1161}
1162
1163
1164/**
1165 * Extracts the CPU stepping from ASMCpuId(1) or ASMCpuId(0x80000001)
1166 *
1167 * @returns Model.
1168 * @param uEAX EAX from ASMCpuId(1) or ASMCpuId(0x80000001).
1169 */
1170DECLINLINE(uint32_t) ASMGetCpuStepping(uint32_t uEAX)
1171{
1172 return uEAX & 0xf;
1173}
1174
1175
1176/**
1177 * Get cr0.
1178 * @returns cr0.
1179 */
1180#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1181DECLASM(RTCCUINTREG) ASMGetCR0(void);
1182#else
1183DECLINLINE(RTCCUINTREG) ASMGetCR0(void)
1184{
1185 RTCCUINTREG uCR0;
1186# if RT_INLINE_ASM_USES_INTRIN
1187 uCR0 = __readcr0();
1188
1189# elif RT_INLINE_ASM_GNU_STYLE
1190# ifdef RT_ARCH_AMD64
1191 __asm__ __volatile__("movq %%cr0, %0\t\n" : "=r" (uCR0));
1192# else
1193 __asm__ __volatile__("movl %%cr0, %0\t\n" : "=r" (uCR0));
1194# endif
1195# else
1196 __asm
1197 {
1198# ifdef RT_ARCH_AMD64
1199 mov rax, cr0
1200 mov [uCR0], rax
1201# else
1202 mov eax, cr0
1203 mov [uCR0], eax
1204# endif
1205 }
1206# endif
1207 return uCR0;
1208}
1209#endif
1210
1211
1212/**
1213 * Sets the CR0 register.
1214 * @param uCR0 The new CR0 value.
1215 */
1216#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1217DECLASM(void) ASMSetCR0(RTCCUINTREG uCR0);
1218#else
1219DECLINLINE(void) ASMSetCR0(RTCCUINTREG uCR0)
1220{
1221# if RT_INLINE_ASM_USES_INTRIN
1222 __writecr0(uCR0);
1223
1224# elif RT_INLINE_ASM_GNU_STYLE
1225# ifdef RT_ARCH_AMD64
1226 __asm__ __volatile__("movq %0, %%cr0\n\t" :: "r" (uCR0));
1227# else
1228 __asm__ __volatile__("movl %0, %%cr0\n\t" :: "r" (uCR0));
1229# endif
1230# else
1231 __asm
1232 {
1233# ifdef RT_ARCH_AMD64
1234 mov rax, [uCR0]
1235 mov cr0, rax
1236# else
1237 mov eax, [uCR0]
1238 mov cr0, eax
1239# endif
1240 }
1241# endif
1242}
1243#endif
1244
1245
1246/**
1247 * Get cr2.
1248 * @returns cr2.
1249 */
1250#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1251DECLASM(RTCCUINTREG) ASMGetCR2(void);
1252#else
1253DECLINLINE(RTCCUINTREG) ASMGetCR2(void)
1254{
1255 RTCCUINTREG uCR2;
1256# if RT_INLINE_ASM_USES_INTRIN
1257 uCR2 = __readcr2();
1258
1259# elif RT_INLINE_ASM_GNU_STYLE
1260# ifdef RT_ARCH_AMD64
1261 __asm__ __volatile__("movq %%cr2, %0\t\n" : "=r" (uCR2));
1262# else
1263 __asm__ __volatile__("movl %%cr2, %0\t\n" : "=r" (uCR2));
1264# endif
1265# else
1266 __asm
1267 {
1268# ifdef RT_ARCH_AMD64
1269 mov rax, cr2
1270 mov [uCR2], rax
1271# else
1272 mov eax, cr2
1273 mov [uCR2], eax
1274# endif
1275 }
1276# endif
1277 return uCR2;
1278}
1279#endif
1280
1281
1282/**
1283 * Sets the CR2 register.
1284 * @param uCR2 The new CR0 value.
1285 */
1286#if RT_INLINE_ASM_EXTERNAL
1287DECLASM(void) ASMSetCR2(RTCCUINTREG uCR2);
1288#else
1289DECLINLINE(void) ASMSetCR2(RTCCUINTREG uCR2)
1290{
1291# if RT_INLINE_ASM_GNU_STYLE
1292# ifdef RT_ARCH_AMD64
1293 __asm__ __volatile__("movq %0, %%cr2\n\t" :: "r" (uCR2));
1294# else
1295 __asm__ __volatile__("movl %0, %%cr2\n\t" :: "r" (uCR2));
1296# endif
1297# else
1298 __asm
1299 {
1300# ifdef RT_ARCH_AMD64
1301 mov rax, [uCR2]
1302 mov cr2, rax
1303# else
1304 mov eax, [uCR2]
1305 mov cr2, eax
1306# endif
1307 }
1308# endif
1309}
1310#endif
1311
1312
1313/**
1314 * Get cr3.
1315 * @returns cr3.
1316 */
1317#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1318DECLASM(RTCCUINTREG) ASMGetCR3(void);
1319#else
1320DECLINLINE(RTCCUINTREG) ASMGetCR3(void)
1321{
1322 RTCCUINTREG uCR3;
1323# if RT_INLINE_ASM_USES_INTRIN
1324 uCR3 = __readcr3();
1325
1326# elif RT_INLINE_ASM_GNU_STYLE
1327# ifdef RT_ARCH_AMD64
1328 __asm__ __volatile__("movq %%cr3, %0\t\n" : "=r" (uCR3));
1329# else
1330 __asm__ __volatile__("movl %%cr3, %0\t\n" : "=r" (uCR3));
1331# endif
1332# else
1333 __asm
1334 {
1335# ifdef RT_ARCH_AMD64
1336 mov rax, cr3
1337 mov [uCR3], rax
1338# else
1339 mov eax, cr3
1340 mov [uCR3], eax
1341# endif
1342 }
1343# endif
1344 return uCR3;
1345}
1346#endif
1347
1348
1349/**
1350 * Sets the CR3 register.
1351 *
1352 * @param uCR3 New CR3 value.
1353 */
1354#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1355DECLASM(void) ASMSetCR3(RTCCUINTREG uCR3);
1356#else
1357DECLINLINE(void) ASMSetCR3(RTCCUINTREG uCR3)
1358{
1359# if RT_INLINE_ASM_USES_INTRIN
1360 __writecr3(uCR3);
1361
1362# elif RT_INLINE_ASM_GNU_STYLE
1363# ifdef RT_ARCH_AMD64
1364 __asm__ __volatile__("movq %0, %%cr3\n\t" : : "r" (uCR3));
1365# else
1366 __asm__ __volatile__("movl %0, %%cr3\n\t" : : "r" (uCR3));
1367# endif
1368# else
1369 __asm
1370 {
1371# ifdef RT_ARCH_AMD64
1372 mov rax, [uCR3]
1373 mov cr3, rax
1374# else
1375 mov eax, [uCR3]
1376 mov cr3, eax
1377# endif
1378 }
1379# endif
1380}
1381#endif
1382
1383
1384/**
1385 * Reloads the CR3 register.
1386 */
1387#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1388DECLASM(void) ASMReloadCR3(void);
1389#else
1390DECLINLINE(void) ASMReloadCR3(void)
1391{
1392# if RT_INLINE_ASM_USES_INTRIN
1393 __writecr3(__readcr3());
1394
1395# elif RT_INLINE_ASM_GNU_STYLE
1396 RTCCUINTREG u;
1397# ifdef RT_ARCH_AMD64
1398 __asm__ __volatile__("movq %%cr3, %0\n\t"
1399 "movq %0, %%cr3\n\t"
1400 : "=r" (u));
1401# else
1402 __asm__ __volatile__("movl %%cr3, %0\n\t"
1403 "movl %0, %%cr3\n\t"
1404 : "=r" (u));
1405# endif
1406# else
1407 __asm
1408 {
1409# ifdef RT_ARCH_AMD64
1410 mov rax, cr3
1411 mov cr3, rax
1412# else
1413 mov eax, cr3
1414 mov cr3, eax
1415# endif
1416 }
1417# endif
1418}
1419#endif
1420
1421
1422/**
1423 * Get cr4.
1424 * @returns cr4.
1425 */
1426#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1427DECLASM(RTCCUINTREG) ASMGetCR4(void);
1428#else
1429DECLINLINE(RTCCUINTREG) ASMGetCR4(void)
1430{
1431 RTCCUINTREG uCR4;
1432# if RT_INLINE_ASM_USES_INTRIN
1433 uCR4 = __readcr4();
1434
1435# elif RT_INLINE_ASM_GNU_STYLE
1436# ifdef RT_ARCH_AMD64
1437 __asm__ __volatile__("movq %%cr4, %0\t\n" : "=r" (uCR4));
1438# else
1439 __asm__ __volatile__("movl %%cr4, %0\t\n" : "=r" (uCR4));
1440# endif
1441# else
1442 __asm
1443 {
1444# ifdef RT_ARCH_AMD64
1445 mov rax, cr4
1446 mov [uCR4], rax
1447# else
1448 push eax /* just in case */
1449 /*mov eax, cr4*/
1450 _emit 0x0f
1451 _emit 0x20
1452 _emit 0xe0
1453 mov [uCR4], eax
1454 pop eax
1455# endif
1456 }
1457# endif
1458 return uCR4;
1459}
1460#endif
1461
1462
1463/**
1464 * Sets the CR4 register.
1465 *
1466 * @param uCR4 New CR4 value.
1467 */
1468#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1469DECLASM(void) ASMSetCR4(RTCCUINTREG uCR4);
1470#else
1471DECLINLINE(void) ASMSetCR4(RTCCUINTREG uCR4)
1472{
1473# if RT_INLINE_ASM_USES_INTRIN
1474 __writecr4(uCR4);
1475
1476# elif RT_INLINE_ASM_GNU_STYLE
1477# ifdef RT_ARCH_AMD64
1478 __asm__ __volatile__("movq %0, %%cr4\n\t" : : "r" (uCR4));
1479# else
1480 __asm__ __volatile__("movl %0, %%cr4\n\t" : : "r" (uCR4));
1481# endif
1482# else
1483 __asm
1484 {
1485# ifdef RT_ARCH_AMD64
1486 mov rax, [uCR4]
1487 mov cr4, rax
1488# else
1489 mov eax, [uCR4]
1490 _emit 0x0F
1491 _emit 0x22
1492 _emit 0xE0 /* mov cr4, eax */
1493# endif
1494 }
1495# endif
1496}
1497#endif
1498
1499
1500/**
1501 * Get cr8.
1502 * @returns cr8.
1503 * @remark The lock prefix hack for access from non-64-bit modes is NOT used and 0 is returned.
1504 */
1505#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1506DECLASM(RTCCUINTREG) ASMGetCR8(void);
1507#else
1508DECLINLINE(RTCCUINTREG) ASMGetCR8(void)
1509{
1510# ifdef RT_ARCH_AMD64
1511 RTCCUINTREG uCR8;
1512# if RT_INLINE_ASM_USES_INTRIN
1513 uCR8 = __readcr8();
1514
1515# elif RT_INLINE_ASM_GNU_STYLE
1516 __asm__ __volatile__("movq %%cr8, %0\t\n" : "=r" (uCR8));
1517# else
1518 __asm
1519 {
1520 mov rax, cr8
1521 mov [uCR8], rax
1522 }
1523# endif
1524 return uCR8;
1525# else /* !RT_ARCH_AMD64 */
1526 return 0;
1527# endif /* !RT_ARCH_AMD64 */
1528}
1529#endif
1530
1531
1532/**
1533 * Enables interrupts (EFLAGS.IF).
1534 */
1535#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1536DECLASM(void) ASMIntEnable(void);
1537#else
1538DECLINLINE(void) ASMIntEnable(void)
1539{
1540# if RT_INLINE_ASM_GNU_STYLE
1541 __asm("sti\n");
1542# elif RT_INLINE_ASM_USES_INTRIN
1543 _enable();
1544# else
1545 __asm sti
1546# endif
1547}
1548#endif
1549
1550
1551/**
1552 * Disables interrupts (!EFLAGS.IF).
1553 */
1554#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1555DECLASM(void) ASMIntDisable(void);
1556#else
1557DECLINLINE(void) ASMIntDisable(void)
1558{
1559# if RT_INLINE_ASM_GNU_STYLE
1560 __asm("cli\n");
1561# elif RT_INLINE_ASM_USES_INTRIN
1562 _disable();
1563# else
1564 __asm cli
1565# endif
1566}
1567#endif
1568
1569
1570/**
1571 * Disables interrupts and returns previous xFLAGS.
1572 */
1573#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1574DECLASM(RTCCUINTREG) ASMIntDisableFlags(void);
1575#else
1576DECLINLINE(RTCCUINTREG) ASMIntDisableFlags(void)
1577{
1578 RTCCUINTREG xFlags;
1579# if RT_INLINE_ASM_GNU_STYLE
1580# ifdef RT_ARCH_AMD64
1581 __asm__ __volatile__("pushfq\n\t"
1582 "cli\n\t"
1583 "popq %0\n\t"
1584 : "=r" (xFlags));
1585# else
1586 __asm__ __volatile__("pushfl\n\t"
1587 "cli\n\t"
1588 "popl %0\n\t"
1589 : "=r" (xFlags));
1590# endif
1591# elif RT_INLINE_ASM_USES_INTRIN && !defined(RT_ARCH_X86)
1592 xFlags = ASMGetFlags();
1593 _disable();
1594# else
1595 __asm {
1596 pushfd
1597 cli
1598 pop [xFlags]
1599 }
1600# endif
1601 return xFlags;
1602}
1603#endif
1604
1605
1606/**
1607 * Are interrupts enabled?
1608 *
1609 * @returns true / false.
1610 */
1611DECLINLINE(RTCCUINTREG) ASMIntAreEnabled(void)
1612{
1613 RTCCUINTREG uFlags = ASMGetFlags();
1614 return uFlags & 0x200 /* X86_EFL_IF */ ? true : false;
1615}
1616
1617
1618/**
1619 * Halts the CPU until interrupted.
1620 */
1621#if RT_INLINE_ASM_EXTERNAL
1622DECLASM(void) ASMHalt(void);
1623#else
1624DECLINLINE(void) ASMHalt(void)
1625{
1626# if RT_INLINE_ASM_GNU_STYLE
1627 __asm__ __volatile__("hlt\n\t");
1628# else
1629 __asm {
1630 hlt
1631 }
1632# endif
1633}
1634#endif
1635
1636
1637/**
1638 * Reads a machine specific register.
1639 *
1640 * @returns Register content.
1641 * @param uRegister Register to read.
1642 */
1643#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1644DECLASM(uint64_t) ASMRdMsr(uint32_t uRegister);
1645#else
1646DECLINLINE(uint64_t) ASMRdMsr(uint32_t uRegister)
1647{
1648 RTUINT64U u;
1649# if RT_INLINE_ASM_GNU_STYLE
1650 __asm__ __volatile__("rdmsr\n\t"
1651 : "=a" (u.s.Lo),
1652 "=d" (u.s.Hi)
1653 : "c" (uRegister));
1654
1655# elif RT_INLINE_ASM_USES_INTRIN
1656 u.u = __readmsr(uRegister);
1657
1658# else
1659 __asm
1660 {
1661 mov ecx, [uRegister]
1662 rdmsr
1663 mov [u.s.Lo], eax
1664 mov [u.s.Hi], edx
1665 }
1666# endif
1667
1668 return u.u;
1669}
1670#endif
1671
1672
1673/**
1674 * Writes a machine specific register.
1675 *
1676 * @returns Register content.
1677 * @param uRegister Register to write to.
1678 * @param u64Val Value to write.
1679 */
1680#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1681DECLASM(void) ASMWrMsr(uint32_t uRegister, uint64_t u64Val);
1682#else
1683DECLINLINE(void) ASMWrMsr(uint32_t uRegister, uint64_t u64Val)
1684{
1685 RTUINT64U u;
1686
1687 u.u = u64Val;
1688# if RT_INLINE_ASM_GNU_STYLE
1689 __asm__ __volatile__("wrmsr\n\t"
1690 ::"a" (u.s.Lo),
1691 "d" (u.s.Hi),
1692 "c" (uRegister));
1693
1694# elif RT_INLINE_ASM_USES_INTRIN
1695 __writemsr(uRegister, u.u);
1696
1697# else
1698 __asm
1699 {
1700 mov ecx, [uRegister]
1701 mov edx, [u.s.Hi]
1702 mov eax, [u.s.Lo]
1703 wrmsr
1704 }
1705# endif
1706}
1707#endif
1708
1709
1710/**
1711 * Reads low part of a machine specific register.
1712 *
1713 * @returns Register content.
1714 * @param uRegister Register to read.
1715 */
1716#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1717DECLASM(uint32_t) ASMRdMsr_Low(uint32_t uRegister);
1718#else
1719DECLINLINE(uint32_t) ASMRdMsr_Low(uint32_t uRegister)
1720{
1721 uint32_t u32;
1722# if RT_INLINE_ASM_GNU_STYLE
1723 __asm__ __volatile__("rdmsr\n\t"
1724 : "=a" (u32)
1725 : "c" (uRegister)
1726 : "edx");
1727
1728# elif RT_INLINE_ASM_USES_INTRIN
1729 u32 = (uint32_t)__readmsr(uRegister);
1730
1731#else
1732 __asm
1733 {
1734 mov ecx, [uRegister]
1735 rdmsr
1736 mov [u32], eax
1737 }
1738# endif
1739
1740 return u32;
1741}
1742#endif
1743
1744
1745/**
1746 * Reads high part of a machine specific register.
1747 *
1748 * @returns Register content.
1749 * @param uRegister Register to read.
1750 */
1751#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1752DECLASM(uint32_t) ASMRdMsr_High(uint32_t uRegister);
1753#else
1754DECLINLINE(uint32_t) ASMRdMsr_High(uint32_t uRegister)
1755{
1756 uint32_t u32;
1757# if RT_INLINE_ASM_GNU_STYLE
1758 __asm__ __volatile__("rdmsr\n\t"
1759 : "=d" (u32)
1760 : "c" (uRegister)
1761 : "eax");
1762
1763# elif RT_INLINE_ASM_USES_INTRIN
1764 u32 = (uint32_t)(__readmsr(uRegister) >> 32);
1765
1766# else
1767 __asm
1768 {
1769 mov ecx, [uRegister]
1770 rdmsr
1771 mov [u32], edx
1772 }
1773# endif
1774
1775 return u32;
1776}
1777#endif
1778
1779
1780/**
1781 * Gets dr0.
1782 *
1783 * @returns dr0.
1784 */
1785#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1786DECLASM(RTCCUINTREG) ASMGetDR0(void);
1787#else
1788DECLINLINE(RTCCUINTREG) ASMGetDR0(void)
1789{
1790 RTCCUINTREG uDR0;
1791# if RT_INLINE_ASM_USES_INTRIN
1792 uDR0 = __readdr(0);
1793# elif RT_INLINE_ASM_GNU_STYLE
1794# ifdef RT_ARCH_AMD64
1795 __asm__ __volatile__("movq %%dr0, %0\n\t" : "=r" (uDR0));
1796# else
1797 __asm__ __volatile__("movl %%dr0, %0\n\t" : "=r" (uDR0));
1798# endif
1799# else
1800 __asm
1801 {
1802# ifdef RT_ARCH_AMD64
1803 mov rax, dr0
1804 mov [uDR0], rax
1805# else
1806 mov eax, dr0
1807 mov [uDR0], eax
1808# endif
1809 }
1810# endif
1811 return uDR0;
1812}
1813#endif
1814
1815
1816/**
1817 * Gets dr1.
1818 *
1819 * @returns dr1.
1820 */
1821#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1822DECLASM(RTCCUINTREG) ASMGetDR1(void);
1823#else
1824DECLINLINE(RTCCUINTREG) ASMGetDR1(void)
1825{
1826 RTCCUINTREG uDR1;
1827# if RT_INLINE_ASM_USES_INTRIN
1828 uDR1 = __readdr(1);
1829# elif RT_INLINE_ASM_GNU_STYLE
1830# ifdef RT_ARCH_AMD64
1831 __asm__ __volatile__("movq %%dr1, %0\n\t" : "=r" (uDR1));
1832# else
1833 __asm__ __volatile__("movl %%dr1, %0\n\t" : "=r" (uDR1));
1834# endif
1835# else
1836 __asm
1837 {
1838# ifdef RT_ARCH_AMD64
1839 mov rax, dr1
1840 mov [uDR1], rax
1841# else
1842 mov eax, dr1
1843 mov [uDR1], eax
1844# endif
1845 }
1846# endif
1847 return uDR1;
1848}
1849#endif
1850
1851
1852/**
1853 * Gets dr2.
1854 *
1855 * @returns dr2.
1856 */
1857#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1858DECLASM(RTCCUINTREG) ASMGetDR2(void);
1859#else
1860DECLINLINE(RTCCUINTREG) ASMGetDR2(void)
1861{
1862 RTCCUINTREG uDR2;
1863# if RT_INLINE_ASM_USES_INTRIN
1864 uDR2 = __readdr(2);
1865# elif RT_INLINE_ASM_GNU_STYLE
1866# ifdef RT_ARCH_AMD64
1867 __asm__ __volatile__("movq %%dr2, %0\n\t" : "=r" (uDR2));
1868# else
1869 __asm__ __volatile__("movl %%dr2, %0\n\t" : "=r" (uDR2));
1870# endif
1871# else
1872 __asm
1873 {
1874# ifdef RT_ARCH_AMD64
1875 mov rax, dr2
1876 mov [uDR2], rax
1877# else
1878 mov eax, dr2
1879 mov [uDR2], eax
1880# endif
1881 }
1882# endif
1883 return uDR2;
1884}
1885#endif
1886
1887
1888/**
1889 * Gets dr3.
1890 *
1891 * @returns dr3.
1892 */
1893#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1894DECLASM(RTCCUINTREG) ASMGetDR3(void);
1895#else
1896DECLINLINE(RTCCUINTREG) ASMGetDR3(void)
1897{
1898 RTCCUINTREG uDR3;
1899# if RT_INLINE_ASM_USES_INTRIN
1900 uDR3 = __readdr(3);
1901# elif RT_INLINE_ASM_GNU_STYLE
1902# ifdef RT_ARCH_AMD64
1903 __asm__ __volatile__("movq %%dr3, %0\n\t" : "=r" (uDR3));
1904# else
1905 __asm__ __volatile__("movl %%dr3, %0\n\t" : "=r" (uDR3));
1906# endif
1907# else
1908 __asm
1909 {
1910# ifdef RT_ARCH_AMD64
1911 mov rax, dr3
1912 mov [uDR3], rax
1913# else
1914 mov eax, dr3
1915 mov [uDR3], eax
1916# endif
1917 }
1918# endif
1919 return uDR3;
1920}
1921#endif
1922
1923
1924/**
1925 * Gets dr6.
1926 *
1927 * @returns dr6.
1928 */
1929#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1930DECLASM(RTCCUINTREG) ASMGetDR6(void);
1931#else
1932DECLINLINE(RTCCUINTREG) ASMGetDR6(void)
1933{
1934 RTCCUINTREG uDR6;
1935# if RT_INLINE_ASM_USES_INTRIN
1936 uDR6 = __readdr(6);
1937# elif RT_INLINE_ASM_GNU_STYLE
1938# ifdef RT_ARCH_AMD64
1939 __asm__ __volatile__("movq %%dr6, %0\n\t" : "=r" (uDR6));
1940# else
1941 __asm__ __volatile__("movl %%dr6, %0\n\t" : "=r" (uDR6));
1942# endif
1943# else
1944 __asm
1945 {
1946# ifdef RT_ARCH_AMD64
1947 mov rax, dr6
1948 mov [uDR6], rax
1949# else
1950 mov eax, dr6
1951 mov [uDR6], eax
1952# endif
1953 }
1954# endif
1955 return uDR6;
1956}
1957#endif
1958
1959
1960/**
1961 * Reads and clears DR6.
1962 *
1963 * @returns DR6.
1964 */
1965#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
1966DECLASM(RTCCUINTREG) ASMGetAndClearDR6(void);
1967#else
1968DECLINLINE(RTCCUINTREG) ASMGetAndClearDR6(void)
1969{
1970 RTCCUINTREG uDR6;
1971# if RT_INLINE_ASM_USES_INTRIN
1972 uDR6 = __readdr(6);
1973 __writedr(6, 0xffff0ff0U); /* 31-16 and 4-11 are 1's, 12 and 63-31 are zero. */
1974# elif RT_INLINE_ASM_GNU_STYLE
1975 RTCCUINTREG uNewValue = 0xffff0ff0U;/* 31-16 and 4-11 are 1's, 12 and 63-31 are zero. */
1976# ifdef RT_ARCH_AMD64
1977 __asm__ __volatile__("movq %%dr6, %0\n\t"
1978 "movq %1, %%dr6\n\t"
1979 : "=r" (uDR6)
1980 : "r" (uNewValue));
1981# else
1982 __asm__ __volatile__("movl %%dr6, %0\n\t"
1983 "movl %1, %%dr6\n\t"
1984 : "=r" (uDR6)
1985 : "r" (uNewValue));
1986# endif
1987# else
1988 __asm
1989 {
1990# ifdef RT_ARCH_AMD64
1991 mov rax, dr6
1992 mov [uDR6], rax
1993 mov rcx, rax
1994 mov ecx, 0ffff0ff0h; /* 31-16 and 4-11 are 1's, 12 and 63-31 are zero. */
1995 mov dr6, rcx
1996# else
1997 mov eax, dr6
1998 mov [uDR6], eax
1999 mov ecx, 0ffff0ff0h; /* 31-16 and 4-11 are 1's, 12 is zero. */
2000 mov dr6, ecx
2001# endif
2002 }
2003# endif
2004 return uDR6;
2005}
2006#endif
2007
2008
2009/**
2010 * Gets dr7.
2011 *
2012 * @returns dr7.
2013 */
2014#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2015DECLASM(RTCCUINTREG) ASMGetDR7(void);
2016#else
2017DECLINLINE(RTCCUINTREG) ASMGetDR7(void)
2018{
2019 RTCCUINTREG uDR7;
2020# if RT_INLINE_ASM_USES_INTRIN
2021 uDR7 = __readdr(7);
2022# elif RT_INLINE_ASM_GNU_STYLE
2023# ifdef RT_ARCH_AMD64
2024 __asm__ __volatile__("movq %%dr7, %0\n\t" : "=r" (uDR7));
2025# else
2026 __asm__ __volatile__("movl %%dr7, %0\n\t" : "=r" (uDR7));
2027# endif
2028# else
2029 __asm
2030 {
2031# ifdef RT_ARCH_AMD64
2032 mov rax, dr7
2033 mov [uDR7], rax
2034# else
2035 mov eax, dr7
2036 mov [uDR7], eax
2037# endif
2038 }
2039# endif
2040 return uDR7;
2041}
2042#endif
2043
2044
2045/**
2046 * Sets dr0.
2047 *
2048 * @param uDRVal Debug register value to write
2049 */
2050#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2051DECLASM(void) ASMSetDR0(RTCCUINTREG uDRVal);
2052#else
2053DECLINLINE(void) ASMSetDR0(RTCCUINTREG uDRVal)
2054{
2055# if RT_INLINE_ASM_USES_INTRIN
2056 __writedr(0, uDRVal);
2057# elif RT_INLINE_ASM_GNU_STYLE
2058# ifdef RT_ARCH_AMD64
2059 __asm__ __volatile__("movq %0, %%dr0\n\t" : : "r" (uDRVal));
2060# else
2061 __asm__ __volatile__("movl %0, %%dr0\n\t" : : "r" (uDRVal));
2062# endif
2063# else
2064 __asm
2065 {
2066# ifdef RT_ARCH_AMD64
2067 mov rax, [uDRVal]
2068 mov dr0, rax
2069# else
2070 mov eax, [uDRVal]
2071 mov dr0, eax
2072# endif
2073 }
2074# endif
2075}
2076#endif
2077
2078
2079/**
2080 * Sets dr1.
2081 *
2082 * @param uDRVal Debug register value to write
2083 */
2084#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2085DECLASM(void) ASMSetDR1(RTCCUINTREG uDRVal);
2086#else
2087DECLINLINE(void) ASMSetDR1(RTCCUINTREG uDRVal)
2088{
2089# if RT_INLINE_ASM_USES_INTRIN
2090 __writedr(1, uDRVal);
2091# elif RT_INLINE_ASM_GNU_STYLE
2092# ifdef RT_ARCH_AMD64
2093 __asm__ __volatile__("movq %0, %%dr1\n\t" : : "r" (uDRVal));
2094# else
2095 __asm__ __volatile__("movl %0, %%dr1\n\t" : : "r" (uDRVal));
2096# endif
2097# else
2098 __asm
2099 {
2100# ifdef RT_ARCH_AMD64
2101 mov rax, [uDRVal]
2102 mov dr1, rax
2103# else
2104 mov eax, [uDRVal]
2105 mov dr1, eax
2106# endif
2107 }
2108# endif
2109}
2110#endif
2111
2112
2113/**
2114 * Sets dr2.
2115 *
2116 * @param uDRVal Debug register value to write
2117 */
2118#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2119DECLASM(void) ASMSetDR2(RTCCUINTREG uDRVal);
2120#else
2121DECLINLINE(void) ASMSetDR2(RTCCUINTREG uDRVal)
2122{
2123# if RT_INLINE_ASM_USES_INTRIN
2124 __writedr(2, uDRVal);
2125# elif RT_INLINE_ASM_GNU_STYLE
2126# ifdef RT_ARCH_AMD64
2127 __asm__ __volatile__("movq %0, %%dr2\n\t" : : "r" (uDRVal));
2128# else
2129 __asm__ __volatile__("movl %0, %%dr2\n\t" : : "r" (uDRVal));
2130# endif
2131# else
2132 __asm
2133 {
2134# ifdef RT_ARCH_AMD64
2135 mov rax, [uDRVal]
2136 mov dr2, rax
2137# else
2138 mov eax, [uDRVal]
2139 mov dr2, eax
2140# endif
2141 }
2142# endif
2143}
2144#endif
2145
2146
2147/**
2148 * Sets dr3.
2149 *
2150 * @param uDRVal Debug register value to write
2151 */
2152#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2153DECLASM(void) ASMSetDR3(RTCCUINTREG uDRVal);
2154#else
2155DECLINLINE(void) ASMSetDR3(RTCCUINTREG uDRVal)
2156{
2157# if RT_INLINE_ASM_USES_INTRIN
2158 __writedr(3, uDRVal);
2159# elif RT_INLINE_ASM_GNU_STYLE
2160# ifdef RT_ARCH_AMD64
2161 __asm__ __volatile__("movq %0, %%dr3\n\t" : : "r" (uDRVal));
2162# else
2163 __asm__ __volatile__("movl %0, %%dr3\n\t" : : "r" (uDRVal));
2164# endif
2165# else
2166 __asm
2167 {
2168# ifdef RT_ARCH_AMD64
2169 mov rax, [uDRVal]
2170 mov dr3, rax
2171# else
2172 mov eax, [uDRVal]
2173 mov dr3, eax
2174# endif
2175 }
2176# endif
2177}
2178#endif
2179
2180
2181/**
2182 * Sets dr6.
2183 *
2184 * @param uDRVal Debug register value to write
2185 */
2186#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2187DECLASM(void) ASMSetDR6(RTCCUINTREG uDRVal);
2188#else
2189DECLINLINE(void) ASMSetDR6(RTCCUINTREG uDRVal)
2190{
2191# if RT_INLINE_ASM_USES_INTRIN
2192 __writedr(6, uDRVal);
2193# elif RT_INLINE_ASM_GNU_STYLE
2194# ifdef RT_ARCH_AMD64
2195 __asm__ __volatile__("movq %0, %%dr6\n\t" : : "r" (uDRVal));
2196# else
2197 __asm__ __volatile__("movl %0, %%dr6\n\t" : : "r" (uDRVal));
2198# endif
2199# else
2200 __asm
2201 {
2202# ifdef RT_ARCH_AMD64
2203 mov rax, [uDRVal]
2204 mov dr6, rax
2205# else
2206 mov eax, [uDRVal]
2207 mov dr6, eax
2208# endif
2209 }
2210# endif
2211}
2212#endif
2213
2214
2215/**
2216 * Sets dr7.
2217 *
2218 * @param uDRVal Debug register value to write
2219 */
2220#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2221DECLASM(void) ASMSetDR7(RTCCUINTREG uDRVal);
2222#else
2223DECLINLINE(void) ASMSetDR7(RTCCUINTREG uDRVal)
2224{
2225# if RT_INLINE_ASM_USES_INTRIN
2226 __writedr(7, uDRVal);
2227# elif RT_INLINE_ASM_GNU_STYLE
2228# ifdef RT_ARCH_AMD64
2229 __asm__ __volatile__("movq %0, %%dr7\n\t" : : "r" (uDRVal));
2230# else
2231 __asm__ __volatile__("movl %0, %%dr7\n\t" : : "r" (uDRVal));
2232# endif
2233# else
2234 __asm
2235 {
2236# ifdef RT_ARCH_AMD64
2237 mov rax, [uDRVal]
2238 mov dr7, rax
2239# else
2240 mov eax, [uDRVal]
2241 mov dr7, eax
2242# endif
2243 }
2244# endif
2245}
2246#endif
2247
2248
2249/**
2250 * Writes a 8-bit unsigned integer to an I/O port, ordered.
2251 *
2252 * @param Port I/O port to write to.
2253 * @param u8 8-bit integer to write.
2254 */
2255#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2256DECLASM(void) ASMOutU8(RTIOPORT Port, uint8_t u8);
2257#else
2258DECLINLINE(void) ASMOutU8(RTIOPORT Port, uint8_t u8)
2259{
2260# if RT_INLINE_ASM_GNU_STYLE
2261 __asm__ __volatile__("outb %b1, %w0\n\t"
2262 :: "Nd" (Port),
2263 "a" (u8));
2264
2265# elif RT_INLINE_ASM_USES_INTRIN
2266 __outbyte(Port, u8);
2267
2268# else
2269 __asm
2270 {
2271 mov dx, [Port]
2272 mov al, [u8]
2273 out dx, al
2274 }
2275# endif
2276}
2277#endif
2278
2279
2280/**
2281 * Reads a 8-bit unsigned integer from an I/O port, ordered.
2282 *
2283 * @returns 8-bit integer.
2284 * @param Port I/O port to read from.
2285 */
2286#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2287DECLASM(uint8_t) ASMInU8(RTIOPORT Port);
2288#else
2289DECLINLINE(uint8_t) ASMInU8(RTIOPORT Port)
2290{
2291 uint8_t u8;
2292# if RT_INLINE_ASM_GNU_STYLE
2293 __asm__ __volatile__("inb %w1, %b0\n\t"
2294 : "=a" (u8)
2295 : "Nd" (Port));
2296
2297# elif RT_INLINE_ASM_USES_INTRIN
2298 u8 = __inbyte(Port);
2299
2300# else
2301 __asm
2302 {
2303 mov dx, [Port]
2304 in al, dx
2305 mov [u8], al
2306 }
2307# endif
2308 return u8;
2309}
2310#endif
2311
2312
2313/**
2314 * Writes a 16-bit unsigned integer to an I/O port, ordered.
2315 *
2316 * @param Port I/O port to write to.
2317 * @param u16 16-bit integer to write.
2318 */
2319#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2320DECLASM(void) ASMOutU16(RTIOPORT Port, uint16_t u16);
2321#else
2322DECLINLINE(void) ASMOutU16(RTIOPORT Port, uint16_t u16)
2323{
2324# if RT_INLINE_ASM_GNU_STYLE
2325 __asm__ __volatile__("outw %w1, %w0\n\t"
2326 :: "Nd" (Port),
2327 "a" (u16));
2328
2329# elif RT_INLINE_ASM_USES_INTRIN
2330 __outword(Port, u16);
2331
2332# else
2333 __asm
2334 {
2335 mov dx, [Port]
2336 mov ax, [u16]
2337 out dx, ax
2338 }
2339# endif
2340}
2341#endif
2342
2343
2344/**
2345 * Reads a 16-bit unsigned integer from an I/O port, ordered.
2346 *
2347 * @returns 16-bit integer.
2348 * @param Port I/O port to read from.
2349 */
2350#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2351DECLASM(uint16_t) ASMInU16(RTIOPORT Port);
2352#else
2353DECLINLINE(uint16_t) ASMInU16(RTIOPORT Port)
2354{
2355 uint16_t u16;
2356# if RT_INLINE_ASM_GNU_STYLE
2357 __asm__ __volatile__("inw %w1, %w0\n\t"
2358 : "=a" (u16)
2359 : "Nd" (Port));
2360
2361# elif RT_INLINE_ASM_USES_INTRIN
2362 u16 = __inword(Port);
2363
2364# else
2365 __asm
2366 {
2367 mov dx, [Port]
2368 in ax, dx
2369 mov [u16], ax
2370 }
2371# endif
2372 return u16;
2373}
2374#endif
2375
2376
2377/**
2378 * Writes a 32-bit unsigned integer to an I/O port, ordered.
2379 *
2380 * @param Port I/O port to write to.
2381 * @param u32 32-bit integer to write.
2382 */
2383#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2384DECLASM(void) ASMOutU32(RTIOPORT Port, uint32_t u32);
2385#else
2386DECLINLINE(void) ASMOutU32(RTIOPORT Port, uint32_t u32)
2387{
2388# if RT_INLINE_ASM_GNU_STYLE
2389 __asm__ __volatile__("outl %1, %w0\n\t"
2390 :: "Nd" (Port),
2391 "a" (u32));
2392
2393# elif RT_INLINE_ASM_USES_INTRIN
2394 __outdword(Port, u32);
2395
2396# else
2397 __asm
2398 {
2399 mov dx, [Port]
2400 mov eax, [u32]
2401 out dx, eax
2402 }
2403# endif
2404}
2405#endif
2406
2407
2408/**
2409 * Reads a 32-bit unsigned integer from an I/O port, ordered.
2410 *
2411 * @returns 32-bit integer.
2412 * @param Port I/O port to read from.
2413 */
2414#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2415DECLASM(uint32_t) ASMInU32(RTIOPORT Port);
2416#else
2417DECLINLINE(uint32_t) ASMInU32(RTIOPORT Port)
2418{
2419 uint32_t u32;
2420# if RT_INLINE_ASM_GNU_STYLE
2421 __asm__ __volatile__("inl %w1, %0\n\t"
2422 : "=a" (u32)
2423 : "Nd" (Port));
2424
2425# elif RT_INLINE_ASM_USES_INTRIN
2426 u32 = __indword(Port);
2427
2428# else
2429 __asm
2430 {
2431 mov dx, [Port]
2432 in eax, dx
2433 mov [u32], eax
2434 }
2435# endif
2436 return u32;
2437}
2438#endif
2439
2440
2441/**
2442 * Writes a string of 8-bit unsigned integer items to an I/O port, ordered.
2443 *
2444 * @param Port I/O port to write to.
2445 * @param pau8 Pointer to the string buffer.
2446 * @param c The number of items to write.
2447 */
2448#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2449DECLASM(void) ASMOutStrU8(RTIOPORT Port, uint8_t const *pau8, size_t c);
2450#else
2451DECLINLINE(void) ASMOutStrU8(RTIOPORT Port, uint8_t const *pau8, size_t c)
2452{
2453# if RT_INLINE_ASM_GNU_STYLE
2454 __asm__ __volatile__("rep; outsb\n\t"
2455 : "+S" (pau8),
2456 "+c" (c)
2457 : "d" (Port));
2458
2459# elif RT_INLINE_ASM_USES_INTRIN
2460 __outbytestring(Port, (unsigned char *)pau8, (unsigned long)c);
2461
2462# else
2463 __asm
2464 {
2465 mov dx, [Port]
2466 mov ecx, [c]
2467 mov eax, [pau8]
2468 xchg esi, eax
2469 rep outsb
2470 xchg esi, eax
2471 }
2472# endif
2473}
2474#endif
2475
2476
2477/**
2478 * Reads a string of 8-bit unsigned integer items from an I/O port, ordered.
2479 *
2480 * @param Port I/O port to read from.
2481 * @param pau8 Pointer to the string buffer (output).
2482 * @param c The number of items to read.
2483 */
2484#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2485DECLASM(void) ASMInStrU8(RTIOPORT Port, uint8_t *pau8, size_t c);
2486#else
2487DECLINLINE(void) ASMInStrU8(RTIOPORT Port, uint8_t *pau8, size_t c)
2488{
2489# if RT_INLINE_ASM_GNU_STYLE
2490 __asm__ __volatile__("rep; insb\n\t"
2491 : "+D" (pau8),
2492 "+c" (c)
2493 : "d" (Port));
2494
2495# elif RT_INLINE_ASM_USES_INTRIN
2496 __inbytestring(Port, pau8, (unsigned long)c);
2497
2498# else
2499 __asm
2500 {
2501 mov dx, [Port]
2502 mov ecx, [c]
2503 mov eax, [pau8]
2504 xchg edi, eax
2505 rep insb
2506 xchg edi, eax
2507 }
2508# endif
2509}
2510#endif
2511
2512
2513/**
2514 * Writes a string of 16-bit unsigned integer items to an I/O port, ordered.
2515 *
2516 * @param Port I/O port to write to.
2517 * @param pau16 Pointer to the string buffer.
2518 * @param c The number of items to write.
2519 */
2520#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2521DECLASM(void) ASMOutStrU16(RTIOPORT Port, uint16_t const *pau16, size_t c);
2522#else
2523DECLINLINE(void) ASMOutStrU16(RTIOPORT Port, uint16_t const *pau16, size_t c)
2524{
2525# if RT_INLINE_ASM_GNU_STYLE
2526 __asm__ __volatile__("rep; outsw\n\t"
2527 : "+S" (pau16),
2528 "+c" (c)
2529 : "d" (Port));
2530
2531# elif RT_INLINE_ASM_USES_INTRIN
2532 __outwordstring(Port, (unsigned short *)pau16, (unsigned long)c);
2533
2534# else
2535 __asm
2536 {
2537 mov dx, [Port]
2538 mov ecx, [c]
2539 mov eax, [pau16]
2540 xchg esi, eax
2541 rep outsw
2542 xchg esi, eax
2543 }
2544# endif
2545}
2546#endif
2547
2548
2549/**
2550 * Reads a string of 16-bit unsigned integer items from an I/O port, ordered.
2551 *
2552 * @param Port I/O port to read from.
2553 * @param pau16 Pointer to the string buffer (output).
2554 * @param c The number of items to read.
2555 */
2556#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2557DECLASM(void) ASMInStrU16(RTIOPORT Port, uint16_t *pau16, size_t c);
2558#else
2559DECLINLINE(void) ASMInStrU16(RTIOPORT Port, uint16_t *pau16, size_t c)
2560{
2561# if RT_INLINE_ASM_GNU_STYLE
2562 __asm__ __volatile__("rep; insw\n\t"
2563 : "+D" (pau16),
2564 "+c" (c)
2565 : "d" (Port));
2566
2567# elif RT_INLINE_ASM_USES_INTRIN
2568 __inwordstring(Port, pau16, (unsigned long)c);
2569
2570# else
2571 __asm
2572 {
2573 mov dx, [Port]
2574 mov ecx, [c]
2575 mov eax, [pau16]
2576 xchg edi, eax
2577 rep insw
2578 xchg edi, eax
2579 }
2580# endif
2581}
2582#endif
2583
2584
2585/**
2586 * Writes a string of 32-bit unsigned integer items to an I/O port, ordered.
2587 *
2588 * @param Port I/O port to write to.
2589 * @param pau32 Pointer to the string buffer.
2590 * @param c The number of items to write.
2591 */
2592#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2593DECLASM(void) ASMOutStrU32(RTIOPORT Port, uint32_t const *pau32, size_t c);
2594#else
2595DECLINLINE(void) ASMOutStrU32(RTIOPORT Port, uint32_t const *pau32, size_t c)
2596{
2597# if RT_INLINE_ASM_GNU_STYLE
2598 __asm__ __volatile__("rep; outsl\n\t"
2599 : "+S" (pau32),
2600 "+c" (c)
2601 : "d" (Port));
2602
2603# elif RT_INLINE_ASM_USES_INTRIN
2604 __outdwordstring(Port, (unsigned long *)pau32, (unsigned long)c);
2605
2606# else
2607 __asm
2608 {
2609 mov dx, [Port]
2610 mov ecx, [c]
2611 mov eax, [pau32]
2612 xchg esi, eax
2613 rep outsd
2614 xchg esi, eax
2615 }
2616# endif
2617}
2618#endif
2619
2620
2621/**
2622 * Reads a string of 32-bit unsigned integer items from an I/O port, ordered.
2623 *
2624 * @param Port I/O port to read from.
2625 * @param pau32 Pointer to the string buffer (output).
2626 * @param c The number of items to read.
2627 */
2628#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2629DECLASM(void) ASMInStrU32(RTIOPORT Port, uint32_t *pau32, size_t c);
2630#else
2631DECLINLINE(void) ASMInStrU32(RTIOPORT Port, uint32_t *pau32, size_t c)
2632{
2633# if RT_INLINE_ASM_GNU_STYLE
2634 __asm__ __volatile__("rep; insl\n\t"
2635 : "+D" (pau32),
2636 "+c" (c)
2637 : "d" (Port));
2638
2639# elif RT_INLINE_ASM_USES_INTRIN
2640 __indwordstring(Port, (unsigned long *)pau32, (unsigned long)c);
2641
2642# else
2643 __asm
2644 {
2645 mov dx, [Port]
2646 mov ecx, [c]
2647 mov eax, [pau32]
2648 xchg edi, eax
2649 rep insd
2650 xchg edi, eax
2651 }
2652# endif
2653}
2654#endif
2655
2656
2657/**
2658 * Invalidate page.
2659 *
2660 * @param pv Address of the page to invalidate.
2661 */
2662#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2663DECLASM(void) ASMInvalidatePage(void *pv);
2664#else
2665DECLINLINE(void) ASMInvalidatePage(void *pv)
2666{
2667# if RT_INLINE_ASM_USES_INTRIN
2668 __invlpg(pv);
2669
2670# elif RT_INLINE_ASM_GNU_STYLE
2671 __asm__ __volatile__("invlpg %0\n\t"
2672 : : "m" (*(uint8_t *)pv));
2673# else
2674 __asm
2675 {
2676# ifdef RT_ARCH_AMD64
2677 mov rax, [pv]
2678 invlpg [rax]
2679# else
2680 mov eax, [pv]
2681 invlpg [eax]
2682# endif
2683 }
2684# endif
2685}
2686#endif
2687
2688
2689/**
2690 * Write back the internal caches and invalidate them.
2691 */
2692#if RT_INLINE_ASM_EXTERNAL && !RT_INLINE_ASM_USES_INTRIN
2693DECLASM(void) ASMWriteBackAndInvalidateCaches(void);
2694#else
2695DECLINLINE(void) ASMWriteBackAndInvalidateCaches(void)
2696{
2697# if RT_INLINE_ASM_USES_INTRIN
2698 __wbinvd();
2699
2700# elif RT_INLINE_ASM_GNU_STYLE
2701 __asm__ __volatile__("wbinvd");
2702# else
2703 __asm
2704 {
2705 wbinvd
2706 }
2707# endif
2708}
2709#endif
2710
2711
2712/**
2713 * Invalidate internal and (perhaps) external caches without first
2714 * flushing dirty cache lines. Use with extreme care.
2715 */
2716#if RT_INLINE_ASM_EXTERNAL
2717DECLASM(void) ASMInvalidateInternalCaches(void);
2718#else
2719DECLINLINE(void) ASMInvalidateInternalCaches(void)
2720{
2721# if RT_INLINE_ASM_GNU_STYLE
2722 __asm__ __volatile__("invd");
2723# else
2724 __asm
2725 {
2726 invd
2727 }
2728# endif
2729}
2730#endif
2731
2732
2733/**
2734 * Memory load/store fence, waits for any pending writes and reads to complete.
2735 * Requires the X86_CPUID_FEATURE_EDX_SSE2 CPUID bit set.
2736 */
2737DECLINLINE(void) ASMMemoryFenceSSE2(void)
2738{
2739#if RT_INLINE_ASM_GNU_STYLE
2740 __asm__ __volatile__ (".byte 0x0f,0xae,0xf0\n\t");
2741#elif RT_INLINE_ASM_USES_INTRIN
2742 _mm_mfence();
2743#else
2744 __asm
2745 {
2746 _emit 0x0f
2747 _emit 0xae
2748 _emit 0xf0
2749 }
2750#endif
2751}
2752
2753
2754/**
2755 * Memory store fence, waits for any writes to complete.
2756 * Requires the X86_CPUID_FEATURE_EDX_SSE CPUID bit set.
2757 */
2758DECLINLINE(void) ASMWriteFenceSSE(void)
2759{
2760#if RT_INLINE_ASM_GNU_STYLE
2761 __asm__ __volatile__ (".byte 0x0f,0xae,0xf8\n\t");
2762#elif RT_INLINE_ASM_USES_INTRIN
2763 _mm_sfence();
2764#else
2765 __asm
2766 {
2767 _emit 0x0f
2768 _emit 0xae
2769 _emit 0xf8
2770 }
2771#endif
2772}
2773
2774
2775/**
2776 * Memory load fence, waits for any pending reads to complete.
2777 * Requires the X86_CPUID_FEATURE_EDX_SSE2 CPUID bit set.
2778 */
2779DECLINLINE(void) ASMReadFenceSSE2(void)
2780{
2781#if RT_INLINE_ASM_GNU_STYLE
2782 __asm__ __volatile__ (".byte 0x0f,0xae,0xe8\n\t");
2783#elif RT_INLINE_ASM_USES_INTRIN
2784 _mm_lfence();
2785#else
2786 __asm
2787 {
2788 _emit 0x0f
2789 _emit 0xae
2790 _emit 0xe8
2791 }
2792#endif
2793}
2794
2795/** @} */
2796#endif
2797
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