ENDFUNC
///--------------------------------------------------------------------------
+/// Checking extended control registers.
+
+FUNC(dispatch_x86ish_xgetbv)
+ // Call with two arguments: a pointer Z_OUT to 8 bytes of output space, and
+ // a 32-bit integer C. Read the 64-bit value of XCR(C), and store it
+ // at Z_OUT.
+
+#if CPUFAM_X86
+# define Z_OUT edi
+ mov edi, [esp + 4]
+ mov ecx, [esp + 8]
+#endif
+#if CPUFAM_AMD64 && ABI_SYSV
+# define Z_OUT rdi
+ mov ecx, esi
+#endif
+#if CPUFAM_AMD64 && ABI_WIN
+# define Z_OUT r8
+ mov r8, rcx
+ mov ecx, edx
+#endif
+ endprologue
+
+ xgetbv
+ mov [Z_OUT + 0], eax
+ mov [Z_OUT + 4], edx
+
+ ret
+
+#undef Z_OUT
+ENDFUNC
+
+///--------------------------------------------------------------------------
/// Checking `rdrand'.
FUNC(dispatch_x86ish_rdrand)
- // Enter with one argument: a pointer X_OUT to a 32-bit word. Try to
- // generate a random word using `rdrand'. If successful, set *X_OUT
- // to the generated word, and return zero; otherwise, return -1.
+ // Enter with two arguments: a code OP requesting either `rdrand' (0)
+ // or `rdseed' (1), and a pointer X_OUT to a 32-bit word. Try to
+ // generate a random word using the requested instruction'. If
+ // successful, set *X_OUT to the generated word, and return zero;
+ // otherwise, return -1.
#if CPUFAM_X86
+# define OP eax
# define X_OUT edx
# define COUNT ecx
- mov X_OUT, [SP + 4]
+ mov OP, [SP + 4]
+ mov X_OUT, [SP + 8]
#endif
#if CPUFAM_AMD64 && ABI_SYSV
-# define X_OUT rdi
+# define OP edi
+# define X_OUT rsi
# define COUNT ecx
#endif
#if CPUFAM_AMD64 && ABI_WIN
-# define X_OUT rcx
-# define COUNT edx
+# define OP rcx
+# define X_OUT rdx
+# define COUNT r8d
#endif
endprologue
+ cmp OP, 0
mov COUNT, 16 // fairly persistent
+ jne 1f
+
0: rdrand eax
jc 9f
dec COUNT
jnz 0b
+ jmp 8f
+
+1: rdseed eax
+ jc 9f
+ dec COUNT
+ jnz 1b
+ jmp 8f
// Failed to come up with a random value.
- mov eax, -1
+8: mov eax, -1
ret
// Success.