2003-11-18 01:27:06 +00:00
/*
2011-04-08 13:03:26 -07:00
Simple DirectMedia Layer
2020-01-16 20:49:25 -08:00
Copyright ( C ) 1997 - 2020 Sam Lantinga < slouken @ libsdl . org >
2003-11-18 01:27:06 +00:00
2011-04-08 13:03:26 -07:00
This software is provided ' as - is ' , without any express or implied
warranty . In no event will the authors be held liable for any damages
arising from the use of this software .
2003-11-18 01:27:06 +00:00
2011-04-08 13:03:26 -07:00
Permission is granted to anyone to use this software for any purpose ,
including commercial applications , and to alter it and redistribute it
freely , subject to the following restrictions :
2003-11-18 01:27:06 +00:00
2011-04-08 13:03:26 -07:00
1. The origin of this software must not be misrepresented ; you must not
claim that you wrote the original software . If you use this software
in a product , an acknowledgment in the product documentation would be
appreciated but is not required .
2. Altered source versions must be plainly marked as such , and must not be
misrepresented as being the original software .
3. This notice may not be removed or altered from any source distribution .
2003-11-18 01:27:06 +00:00
*/
2006-07-10 21:04:37 +00:00
/**
2009-10-19 13:31:58 +00:00
* \ file SDL_cpuinfo . h
2013-05-18 14:17:52 -07:00
*
2009-10-19 13:31:58 +00:00
* CPU feature detection for SDL .
2006-07-10 21:04:37 +00:00
*/
2003-11-18 01:27:06 +00:00
2016-11-20 21:34:54 -08:00
# ifndef SDL_cpuinfo_h_
# define SDL_cpuinfo_h_
2003-11-18 01:27:06 +00:00
2006-02-21 08:46:50 +00:00
# include "SDL_stdinc.h"
2011-02-22 21:44:36 -08:00
/* Need to do this here because intrin.h has C++ code in it */
/* Visual Studio 2005 has a bug where intrin.h conflicts with winnt.h */
2012-10-27 18:39:09 -04:00
# if defined(_MSC_VER) && (_MSC_VER >= 1500) && (defined(_M_IX86) || defined(_M_X64))
2017-08-19 03:07:44 -07:00
# ifdef __clang__
/* Many of the intrinsics SDL uses are not implemented by clang with Visual Studio */
# undef __MMX__
# undef __SSE__
# undef __SSE2__
# else
2011-02-22 21:44:36 -08:00
# include <intrin.h>
2011-03-07 22:04:10 -08:00
# ifndef _WIN64
2019-04-23 16:57:34 -07:00
# ifndef __MMX__
2011-02-22 21:44:36 -08:00
# define __MMX__
2019-04-23 16:57:34 -07:00
# endif
# ifndef __3dNOW__
2011-02-22 21:44:36 -08:00
# define __3dNOW__
2011-03-07 22:04:10 -08:00
# endif
2019-04-23 16:57:34 -07:00
# endif
# ifndef __SSE__
2011-02-22 21:44:36 -08:00
# define __SSE__
2019-04-23 16:57:34 -07:00
# endif
# ifndef __SSE2__
2011-02-22 21:44:36 -08:00
# define __SSE2__
2019-04-23 16:57:34 -07:00
# endif
2017-08-19 03:07:44 -07:00
# endif /* __clang__ */
2011-02-22 21:44:36 -08:00
# elif defined(__MINGW64_VERSION_MAJOR)
# include <intrin.h>
# else
2018-11-19 21:28:52 -08:00
/* altivec.h redefining bool causes a number of problems, see bugs 3993 and 4392, so you need to explicitly define SDL_ENABLE_ALTIVEC_H to have it included. */
# if defined(HAVE_ALTIVEC_H) && defined(__ALTIVEC__) && !defined(__APPLE_ALTIVEC__) && defined(SDL_ENABLE_ALTIVEC_H)
2011-02-22 22:17:44 -08:00
# include <altivec.h>
# endif
2018-12-04 16:50:31 +01:00
# if !defined(SDL_DISABLE_ARM_NEON_H)
# if defined(__ARM_NEON)
# include <arm_neon.h>
# elif defined(__WINDOWS__) || defined(__WINRT__)
2018-12-04 18:15:45 +01:00
/* Visual Studio doesn't define __ARM_ARCH, but _M_ARM (if set, always 7), and _M_ARM64 (if set, always 1). */
2018-12-04 16:50:31 +01:00
# if defined(_M_ARM)
2018-12-04 18:15:45 +01:00
# include <armintr.h>
2018-12-04 16:50:31 +01:00
# include <arm_neon.h>
2019-06-25 11:47:02 +03:00
# define __ARM_NEON 1 /* Set __ARM_NEON so that it can be used elsewhere, at compile time */
2018-12-04 16:50:31 +01:00
# endif
# if defined (_M_ARM64)
2019-08-08 13:26:05 -07:00
# include <arm64intr.h>
# include <arm64_neon.h>
2019-06-25 11:47:02 +03:00
# define __ARM_NEON 1 /* Set __ARM_NEON so that it can be used elsewhere, at compile time */
2018-12-04 16:50:31 +01:00
# endif
# endif
2018-05-16 02:03:06 -04:00
# endif
2017-12-12 16:25:43 -08:00
# if defined(__3dNOW__) && !defined(SDL_DISABLE_MM3DNOW_H)
2011-02-22 21:44:36 -08:00
# include <mm3dnow.h>
# endif
2018-03-10 21:20:25 -08:00
# if defined(HAVE_IMMINTRIN_H) && !defined(SDL_DISABLE_IMMINTRIN_H)
2017-11-10 14:22:21 -08:00
# include <immintrin.h>
2017-12-11 12:00:12 -08:00
# else
2017-12-12 16:25:43 -08:00
# if defined(__MMX__) && !defined(SDL_DISABLE_MMINTRIN_H)
2017-12-11 12:00:12 -08:00
# include <mmintrin.h>
# endif
2017-12-12 16:25:43 -08:00
# if defined(__SSE__) && !defined(SDL_DISABLE_XMMINTRIN_H)
2017-12-11 12:00:12 -08:00
# include <xmmintrin.h>
# endif
2017-12-12 16:25:43 -08:00
# if defined(__SSE2__) && !defined(SDL_DISABLE_EMMINTRIN_H)
2017-12-11 12:00:12 -08:00
# include <emmintrin.h>
2011-02-22 21:44:36 -08:00
# endif
2017-12-12 16:25:43 -08:00
# if defined(__SSE3__) && !defined(SDL_DISABLE_PMMINTRIN_H)
2017-12-11 12:00:12 -08:00
# include <pmmintrin.h>
2017-11-11 16:20:00 -08:00
# endif
2017-12-11 12:00:12 -08:00
# endif /* HAVE_IMMINTRIN_H */
# endif /* compiler version */
2011-02-22 21:44:36 -08:00
2003-11-18 01:27:06 +00:00
# include "begin_code.h"
/* Set up for C function definitions, even when using C++ */
# ifdef __cplusplus
extern " C " {
# endif
2011-01-27 16:51:47 -08:00
/* This is a guess for the cacheline size used for padding.
* Most x86 processors have a 64 byte cache line .
* The 64 - bit PowerPC processors have a 128 byte cache line .
* We ' ll use the larger value to be generally safe .
2011-01-27 16:46:15 -08:00
*/
2011-01-27 16:51:47 -08:00
# define SDL_CACHELINE_SIZE 128
2011-01-27 16:46:15 -08:00
2009-12-16 06:53:53 +00:00
/**
* This function returns the number of CPU cores available .
*/
extern DECLSPEC int SDLCALL SDL_GetCPUCount ( void ) ;
2011-01-27 20:46:45 -08:00
/**
* This function returns the L1 cache line size of the CPU
*
* This is useful for determining multi - threaded structure padding
* or SIMD prefetch sizes .
*/
extern DECLSPEC int SDLCALL SDL_GetCPUCacheLineSize ( void ) ;
2009-10-19 13:31:58 +00:00
/**
* This function returns true if the CPU has the RDTSC instruction .
2003-11-24 09:16:52 +00:00
*/
2006-02-24 10:23:49 +00:00
extern DECLSPEC SDL_bool SDLCALL SDL_HasRDTSC ( void ) ;
2003-11-24 09:16:52 +00:00
2011-02-22 21:44:36 -08:00
/**
* This function returns true if the CPU has AltiVec features .
*/
extern DECLSPEC SDL_bool SDLCALL SDL_HasAltiVec ( void ) ;
2009-10-19 13:31:58 +00:00
/**
* This function returns true if the CPU has MMX features .
2003-11-18 01:27:06 +00:00
*/
2006-02-24 10:23:49 +00:00
extern DECLSPEC SDL_bool SDLCALL SDL_HasMMX ( void ) ;
2003-11-18 01:27:06 +00:00
2011-02-22 21:44:36 -08:00
/**
* This function returns true if the CPU has 3 DNow ! features .
*/
extern DECLSPEC SDL_bool SDLCALL SDL_Has3DNow ( void ) ;
2009-10-19 13:31:58 +00:00
/**
* This function returns true if the CPU has SSE features .
2003-11-18 01:27:06 +00:00
*/
2006-02-24 10:23:49 +00:00
extern DECLSPEC SDL_bool SDLCALL SDL_HasSSE ( void ) ;
2003-11-18 01:27:06 +00:00
2009-10-19 13:31:58 +00:00
/**
* This function returns true if the CPU has SSE2 features .
2004-01-24 05:47:19 +00:00
*/
2006-02-24 10:23:49 +00:00
extern DECLSPEC SDL_bool SDLCALL SDL_HasSSE2 ( void ) ;
2004-01-24 05:47:19 +00:00
2009-10-19 13:31:58 +00:00
/**
2011-02-11 14:51:04 -08:00
* This function returns true if the CPU has SSE3 features .
*/
extern DECLSPEC SDL_bool SDLCALL SDL_HasSSE3 ( void ) ;
/**
2011-02-11 23:02:35 -08:00
* This function returns true if the CPU has SSE4 .1 features .
2004-01-06 17:18:38 +00:00
*/
2011-02-11 23:02:35 -08:00
extern DECLSPEC SDL_bool SDLCALL SDL_HasSSE41 ( void ) ;
/**
* This function returns true if the CPU has SSE4 .2 features .
*/
extern DECLSPEC SDL_bool SDLCALL SDL_HasSSE42 ( void ) ;
2011-02-11 14:51:04 -08:00
2014-02-02 00:33:31 -08:00
/**
* This function returns true if the CPU has AVX features .
*/
extern DECLSPEC SDL_bool SDLCALL SDL_HasAVX ( void ) ;
2014-07-11 22:02:50 -07:00
/**
* This function returns true if the CPU has AVX2 features .
*/
extern DECLSPEC SDL_bool SDLCALL SDL_HasAVX2 ( void ) ;
2018-05-21 11:35:42 -04:00
/**
* This function returns true if the CPU has AVX - 512F ( foundation ) features .
*/
extern DECLSPEC SDL_bool SDLCALL SDL_HasAVX512F ( void ) ;
2019-10-24 21:12:08 -04:00
/**
* This function returns true if the CPU has ARM SIMD ( ARMv6 ) features .
*/
extern DECLSPEC SDL_bool SDLCALL SDL_HasARMSIMD ( void ) ;
2016-11-17 01:15:16 -05:00
/**
* This function returns true if the CPU has NEON ( ARM SIMD ) features .
*/
extern DECLSPEC SDL_bool SDLCALL SDL_HasNEON ( void ) ;
2013-10-17 11:32:56 -07:00
/**
* This function returns the amount of RAM configured in the system , in MB .
*/
extern DECLSPEC int SDLCALL SDL_GetSystemRAM ( void ) ;
2019-06-08 14:54:37 -07:00
/**
* \ brief Report the alignment this system needs for SIMD allocations .
*
* This will return the minimum number of bytes to which a pointer must be
* aligned to be compatible with SIMD instructions on the current machine .
* For example , if the machine supports SSE only , it will return 16 , but if
* it supports AVX - 512F , it ' ll return 64 ( etc ) . This only reports values for
* instruction sets SDL knows about , so if your SDL build doesn ' t have
* SDL_HasAVX512F ( ) , then it might return 16 for the SSE support it sees and
* not 64 for the AVX - 512 instructions that exist but SDL doesn ' t know about .
* Plan accordingly .
*/
extern DECLSPEC size_t SDLCALL SDL_SIMDGetAlignment ( void ) ;
/**
* \ brief Allocate memory in a SIMD - friendly way .
*
* This will allocate a block of memory that is suitable for use with SIMD
* instructions . Specifically , it will be properly aligned and padded for
* the system ' s supported vector instructions .
*
* The memory returned will be padded such that it is safe to read or write
* an incomplete vector at the end of the memory block . This can be useful
* so you don ' t have to drop back to a scalar fallback at the end of your
* SIMD processing loop to deal with the final elements without overflowing
* the allocated buffer .
*
* You must free this memory with SDL_FreeSIMD ( ) , not free ( ) or SDL_free ( )
* or delete [ ] , etc .
*
* Note that SDL will only deal with SIMD instruction sets it is aware of ;
* for example , SDL 2.0 .8 knows that SSE wants 16 - byte vectors
* ( SDL_HasSSE ( ) ) , and AVX2 wants 32 bytes ( SDL_HasAVX2 ( ) ) , but doesn ' t
* know that AVX - 512 wants 64. To be clear : if you can ' t decide to use an
* instruction set with an SDL_Has * ( ) function , don ' t use that instruction
* set with memory allocated through here .
*
* SDL_AllocSIMD ( 0 ) will return a non - NULL pointer , assuming the system isn ' t
* out of memory .
*
* \ param len The length , in bytes , of the block to allocated . The actual
* allocated block might be larger due to padding , etc .
* \ return Pointer to newly - allocated block , NULL if out of memory .
*
* \ sa SDL_SIMDAlignment
* \ sa SDL_SIMDFree
*/
extern DECLSPEC void * SDLCALL SDL_SIMDAlloc ( const size_t len ) ;
/**
* \ brief Deallocate memory obtained from SDL_SIMDAlloc
*
* It is not valid to use this function on a pointer from anything but
* SDL_SIMDAlloc ( ) . It can ' t be used on pointers from malloc , realloc ,
* SDL_malloc , memalign , new [ ] , etc .
*
* However , SDL_SIMDFree ( NULL ) is a legal no - op .
*
* \ sa SDL_SIMDAlloc
*/
extern DECLSPEC void SDLCALL SDL_SIMDFree ( void * ptr ) ;
/* vi: set ts=4 sw=4 expandtab: */
2003-11-18 01:27:06 +00:00
/* Ends C function definitions when using C++ */
# ifdef __cplusplus
}
# endif
# include "close_code.h"
2016-11-20 21:34:54 -08:00
# endif /* SDL_cpuinfo_h_ */
2006-07-10 21:04:37 +00:00
/* vi: set ts=4 sw=4 expandtab: */