mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-26 11:10:58 +00:00 
			
		
		
		
	- Remove most __ASSEMBLER__ __LINKER__ ifdefs - Rename libc/intrin/bits.h to libc/serialize.h - Block pthread cancelation in fchmodat() polyfill - Remove `clang-format off` statements in third_party
		
			
				
	
	
		
			63 lines
		
	
	
	
		
			3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			63 lines
		
	
	
	
		
			3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #ifndef COSMOPOLITAN_LIBC_NEXGEN32E_RDTSCP_H_
 | |
| #define COSMOPOLITAN_LIBC_NEXGEN32E_RDTSCP_H_
 | |
| #include "libc/dce.h"
 | |
| #include "libc/intrin/asmflag.h"
 | |
| #include "libc/nexgen32e/x86feature.h"
 | |
| COSMOPOLITAN_C_START_
 | |
| 
 | |
| #define TSC_AUX_CORE(MSR) ((MSR)&0xfff)
 | |
| #define TSC_AUX_NODE(MSR) (((MSR) >> 12) & 0xfff)
 | |
| 
 | |
| /**
 | |
|  * Reads CPU timestamp counter and IA32_TSC_AUX.
 | |
|  *
 | |
|  * This macro inhibits compiler magic.
 | |
|  * This macro does not inhibit CPU magic.
 | |
|  *
 | |
|  * @see X86_HAVE(RDTSCP)
 | |
|  */
 | |
| #define rdtscp(OPT_OUT_IA32_TSC_AUX)               \
 | |
|   ({                                               \
 | |
|     uint32_t Ecx, *EcxOut;                         \
 | |
|     uint64_t Rax, Rcx, Rdx;                        \
 | |
|     asm volatile("rdtscp"                          \
 | |
|                  : "=a"(Rax), "=c"(Ecx), "=d"(Rdx) \
 | |
|                  : /* no inputs */                 \
 | |
|                  : "memory");                      \
 | |
|     EcxOut = (OPT_OUT_IA32_TSC_AUX);               \
 | |
|     if (EcxOut) *EcxOut = Ecx;                     \
 | |
|     Rdx << 32 | Rax;                               \
 | |
|   })
 | |
| 
 | |
| /**
 | |
|  * Reads timestamp counter auxiliary model specific register value.
 | |
|  */
 | |
| #define rdpid()                                                           \
 | |
|   ({                                                                      \
 | |
|     long Msr;                                                             \
 | |
|     bool32 Ok;                                                            \
 | |
|     Ok = false;                                                           \
 | |
|     if (X86_HAVE(RDPID)) {                                                \
 | |
|       asm volatile("rdpid\t%0" : "=r"(Msr) : /* no inputs */ : "memory"); \
 | |
|       Ok = true;                                                          \
 | |
|     } else if (IsLinux()) {                                               \
 | |
|       asm volatile(ZFLAG_ASM("lsl\t%2,%1")                                \
 | |
|                    : ZFLAG_CONSTRAINT(Ok), "=r"(Msr)                      \
 | |
|                    : "r"(0x7b)                                            \
 | |
|                    : "memory");                                           \
 | |
|     }                                                                     \
 | |
|     if (!Ok && X86_HAVE(RDTSCP)) {                                        \
 | |
|       asm volatile("rdtscp"                                               \
 | |
|                    : "=c"(Msr)                                            \
 | |
|                    : /* no inputs */                                      \
 | |
|                    : "eax", "edx", "memory");                             \
 | |
|       Ok = true;                                                          \
 | |
|     }                                                                     \
 | |
|     if (!Ok) {                                                            \
 | |
|       Msr = -1;                                                           \
 | |
|     }                                                                     \
 | |
|     Msr;                                                                  \
 | |
|   })
 | |
| 
 | |
| COSMOPOLITAN_C_END_
 | |
| #endif /* COSMOPOLITAN_LIBC_NEXGEN32E_RDTSCP_H_ */
 |