Home | History | Annotate | Download | only in pixman
      1 /*
      2  * Copyright  2000 SuSE, Inc.
      3  * Copyright  2007 Red Hat, Inc.
      4  *
      5  * Permission to use, copy, modify, distribute, and sell this software and its
      6  * documentation for any purpose is hereby granted without fee, provided that
      7  * the above copyright notice appear in all copies and that both that
      8  * copyright notice and this permission notice appear in supporting
      9  * documentation, and that the name of SuSE not be used in advertising or
     10  * publicity pertaining to distribution of the software without specific,
     11  * written prior permission.  SuSE makes no representations about the
     12  * suitability of this software for any purpose.  It is provided "as is"
     13  * without express or implied warranty.
     14  *
     15  * SuSE DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING ALL
     16  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL SuSE
     17  * BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
     18  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
     19  * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
     20  * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
     21  */
     22 #ifdef HAVE_CONFIG_H
     23 #include <config.h>
     24 #endif
     25 
     26 #include "pixman-private.h"
     27 
     28 typedef enum
     29 {
     30     ARM_V7		= (1 << 0),
     31     ARM_V6		= (1 << 1),
     32     ARM_VFP		= (1 << 2),
     33     ARM_NEON		= (1 << 3),
     34     ARM_IWMMXT		= (1 << 4)
     35 } arm_cpu_features_t;
     36 
     37 #if defined(USE_ARM_SIMD) || defined(USE_ARM_NEON) || defined(USE_ARM_IWMMXT)
     38 
     39 #if defined(_MSC_VER)
     40 
     41 /* Needed for EXCEPTION_ILLEGAL_INSTRUCTION */
     42 #include <windows.h>
     43 
     44 extern int pixman_msvc_try_arm_neon_op ();
     45 extern int pixman_msvc_try_arm_simd_op ();
     46 
     47 static arm_cpu_features_t
     48 detect_cpu_features (void)
     49 {
     50     arm_cpu_features_t features = 0;
     51 
     52     __try
     53     {
     54 	pixman_msvc_try_arm_simd_op ();
     55 	features |= ARM_V6;
     56     }
     57     __except (GetExceptionCode () == EXCEPTION_ILLEGAL_INSTRUCTION)
     58     {
     59     }
     60 
     61     __try
     62     {
     63 	pixman_msvc_try_arm_neon_op ();
     64 	features |= ARM_NEON;
     65     }
     66     __except (GetExceptionCode () == EXCEPTION_ILLEGAL_INSTRUCTION)
     67     {
     68     }
     69 
     70     return features;
     71 }
     72 
     73 #elif defined(__APPLE__) && defined(TARGET_OS_IPHONE) /* iOS */
     74 
     75 #include "TargetConditionals.h"
     76 
     77 static arm_cpu_features_t
     78 detect_cpu_features (void)
     79 {
     80     arm_cpu_features_t features = 0;
     81 
     82     features |= ARM_V6;
     83 
     84     /* Detection of ARM NEON on iOS is fairly simple because iOS binaries
     85      * contain separate executable images for each processor architecture.
     86      * So all we have to do is detect the armv7 architecture build. The
     87      * operating system automatically runs the armv7 binary for armv7 devices
     88      * and the armv6 binary for armv6 devices.
     89      */
     90 #if defined(__ARM_NEON__)
     91     features |= ARM_NEON;
     92 #endif
     93 
     94     return features;
     95 }
     96 
     97 #elif defined(__ANDROID__) || defined(ANDROID) /* Android */
     98 
     99 #include <cpu-features.h>
    100 
    101 static arm_cpu_features_t
    102 detect_cpu_features (void)
    103 {
    104     arm_cpu_features_t features = 0;
    105     AndroidCpuFamily cpu_family;
    106     uint64_t cpu_features;
    107 
    108     cpu_family = android_getCpuFamily();
    109     cpu_features = android_getCpuFeatures();
    110 
    111     if (cpu_family == ANDROID_CPU_FAMILY_ARM)
    112     {
    113 	if (cpu_features & ANDROID_CPU_ARM_FEATURE_ARMv7)
    114 	    features |= ARM_V7;
    115 
    116 	if (cpu_features & ANDROID_CPU_ARM_FEATURE_VFPv3)
    117 	    features |= ARM_VFP;
    118 
    119 	if (cpu_features & ANDROID_CPU_ARM_FEATURE_NEON)
    120 	    features |= ARM_NEON;
    121     }
    122 
    123     return features;
    124 }
    125 
    126 #elif defined (__linux__) /* linux ELF */
    127 
    128 #include <unistd.h>
    129 #include <sys/types.h>
    130 #include <sys/stat.h>
    131 #include <sys/mman.h>
    132 #include <fcntl.h>
    133 #include <string.h>
    134 #include <elf.h>
    135 
    136 static arm_cpu_features_t
    137 detect_cpu_features (void)
    138 {
    139     arm_cpu_features_t features = 0;
    140     Elf32_auxv_t aux;
    141     int fd;
    142 
    143     fd = open ("/proc/self/auxv", O_RDONLY);
    144     if (fd >= 0)
    145     {
    146 	while (read (fd, &aux, sizeof(Elf32_auxv_t)) == sizeof(Elf32_auxv_t))
    147 	{
    148 	    if (aux.a_type == AT_HWCAP)
    149 	    {
    150 		uint32_t hwcap = aux.a_un.a_val;
    151 
    152 		/* hardcode these values to avoid depending on specific
    153 		 * versions of the hwcap header, e.g. HWCAP_NEON
    154 		 */
    155 		if ((hwcap & 64) != 0)
    156 		    features |= ARM_VFP;
    157 		if ((hwcap & 512) != 0)
    158 		    features |= ARM_IWMMXT;
    159 		/* this flag is only present on kernel 2.6.29 */
    160 		if ((hwcap & 4096) != 0)
    161 		    features |= ARM_NEON;
    162 	    }
    163 	    else if (aux.a_type == AT_PLATFORM)
    164 	    {
    165 		const char *plat = (const char*) aux.a_un.a_val;
    166 
    167 		if (strncmp (plat, "v7l", 3) == 0)
    168 		    features |= (ARM_V7 | ARM_V6);
    169 		else if (strncmp (plat, "v6l", 3) == 0)
    170 		    features |= ARM_V6;
    171 	    }
    172 	}
    173 	close (fd);
    174     }
    175 
    176     return features;
    177 }
    178 
    179 #else /* Unknown */
    180 
    181 static arm_cpu_features_t
    182 detect_cpu_features (void)
    183 {
    184     return 0;
    185 }
    186 
    187 #endif /* Linux elf */
    188 
    189 static pixman_bool_t
    190 have_feature (arm_cpu_features_t feature)
    191 {
    192     static pixman_bool_t initialized;
    193     static arm_cpu_features_t features;
    194 
    195     if (!initialized)
    196     {
    197 	features = detect_cpu_features();
    198 	initialized = TRUE;
    199     }
    200 
    201     return (features & feature) == feature;
    202 }
    203 
    204 #endif /* USE_ARM_SIMD || USE_ARM_NEON || USE_ARM_IWMMXT */
    205 
    206 pixman_implementation_t *
    207 _pixman_arm_get_implementations (pixman_implementation_t *imp)
    208 {
    209 #ifdef USE_ARM_SIMD
    210     if (!_pixman_disabled ("arm-simd") && have_feature (ARM_V6))
    211 	imp = _pixman_implementation_create_arm_simd (imp);
    212 #endif
    213 
    214 #ifdef USE_ARM_IWMMXT
    215     if (!_pixman_disabled ("arm-iwmmxt") && have_feature (ARM_IWMMXT))
    216 	imp = _pixman_implementation_create_mmx (imp);
    217 #endif
    218 
    219 #ifdef USE_ARM_NEON
    220     if (!_pixman_disabled ("arm-neon") && have_feature (ARM_NEON))
    221 	imp = _pixman_implementation_create_arm_neon (imp);
    222 #endif
    223 
    224     return imp;
    225 }
    226