68 lines
		
	
	
		
			1.3 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			68 lines
		
	
	
		
			1.3 KiB
		
	
	
	
		
			C
		
	
	
	
/* cpuid.h: Define cpuid instruction
 | 
						|
 | 
						|
   Copyright 2003, 2012, 2014 Red Hat, Inc.
 | 
						|
 | 
						|
This file is part of Cygwin.
 | 
						|
 | 
						|
This software is a copyrighted work licensed under the terms of the
 | 
						|
Cygwin license.  Please consult the file "CYGWIN_LICENSE" for
 | 
						|
details. */
 | 
						|
 | 
						|
#ifndef CPUID_H
 | 
						|
#define CPUID_H
 | 
						|
 | 
						|
static inline void __attribute ((always_inline))
 | 
						|
cpuid (uint32_t *a, uint32_t *b, uint32_t *c, uint32_t *d, uint32_t ain,
 | 
						|
       uint32_t cin = 0)
 | 
						|
{
 | 
						|
  asm volatile ("cpuid"
 | 
						|
		: "=a" (*a), "=b" (*b), "=c" (*c), "=d" (*d)
 | 
						|
		: "a" (ain), "c" (cin));
 | 
						|
}
 | 
						|
 | 
						|
#ifdef __x86_64__
 | 
						|
static inline bool __attribute ((always_inline))
 | 
						|
can_set_flag (register uint32_t long flag)
 | 
						|
{
 | 
						|
  register uint32_t long r1, r2;
 | 
						|
 | 
						|
  asm volatile ("pushfq\n"
 | 
						|
		"popq %0\n"
 | 
						|
		"movq %0, %1\n"
 | 
						|
		"xorq %2, %0\n"
 | 
						|
		"pushq %0\n"
 | 
						|
		"popfq\n"
 | 
						|
		"pushfq\n"
 | 
						|
		"popq %0\n"
 | 
						|
		"pushq %1\n"
 | 
						|
		"popfq\n"
 | 
						|
		: "=&r" (r1), "=&r" (r2)
 | 
						|
		: "ir" (flag)
 | 
						|
  );
 | 
						|
  return ((r1 ^ r2) & flag) != 0;
 | 
						|
}
 | 
						|
#else
 | 
						|
static inline bool __attribute ((always_inline))
 | 
						|
can_set_flag (register uint32_t flag)
 | 
						|
{
 | 
						|
  register uint32_t r1, r2;
 | 
						|
 | 
						|
  asm volatile ("pushfl\n"
 | 
						|
		"popl %0\n"
 | 
						|
		"movl %0, %1\n"
 | 
						|
		"xorl %2, %0\n"
 | 
						|
		"pushl %0\n"
 | 
						|
		"popfl\n"
 | 
						|
		"pushfl\n"
 | 
						|
		"popl %0\n"
 | 
						|
		"pushl %1\n"
 | 
						|
		"popfl\n"
 | 
						|
		: "=&r" (r1), "=&r" (r2)
 | 
						|
		: "ir" (flag)
 | 
						|
  );
 | 
						|
  return ((r1 ^ r2) & flag) != 0;
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
#endif // !CPUID_H
 |