Blame pixman/pixman-arm.c

Packit 030a23
/*
Packit 030a23
 * Copyright © 2000 SuSE, Inc.
Packit 030a23
 * Copyright © 2007 Red Hat, Inc.
Packit 030a23
 *
Packit 030a23
 * Permission to use, copy, modify, distribute, and sell this software and its
Packit 030a23
 * documentation for any purpose is hereby granted without fee, provided that
Packit 030a23
 * the above copyright notice appear in all copies and that both that
Packit 030a23
 * copyright notice and this permission notice appear in supporting
Packit 030a23
 * documentation, and that the name of SuSE not be used in advertising or
Packit 030a23
 * publicity pertaining to distribution of the software without specific,
Packit 030a23
 * written prior permission.  SuSE makes no representations about the
Packit 030a23
 * suitability of this software for any purpose.  It is provided "as is"
Packit 030a23
 * without express or implied warranty.
Packit 030a23
 *
Packit 030a23
 * SuSE DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING ALL
Packit 030a23
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL SuSE
Packit 030a23
 * BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
Packit 030a23
 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
Packit 030a23
 * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
Packit 030a23
 * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
Packit 030a23
 */
Packit 030a23
#ifdef HAVE_CONFIG_H
Packit 030a23
#include <config.h>
Packit 030a23
#endif
Packit 030a23
Packit 030a23
#include "pixman-private.h"
Packit 030a23
Packit 030a23
typedef enum
Packit 030a23
{
Packit 030a23
    ARM_V7		= (1 << 0),
Packit 030a23
    ARM_V6		= (1 << 1),
Packit 030a23
    ARM_VFP		= (1 << 2),
Packit 030a23
    ARM_NEON		= (1 << 3),
Packit 030a23
    ARM_IWMMXT		= (1 << 4)
Packit 030a23
} arm_cpu_features_t;
Packit 030a23
Packit 030a23
#if defined(USE_ARM_SIMD) || defined(USE_ARM_NEON) || defined(USE_ARM_IWMMXT)
Packit 030a23
Packit 030a23
#if defined(_MSC_VER)
Packit 030a23
Packit 030a23
/* Needed for EXCEPTION_ILLEGAL_INSTRUCTION */
Packit 030a23
#include <windows.h>
Packit 030a23
Packit 030a23
extern int pixman_msvc_try_arm_neon_op ();
Packit 030a23
extern int pixman_msvc_try_arm_simd_op ();
Packit 030a23
Packit 030a23
static arm_cpu_features_t
Packit 030a23
detect_cpu_features (void)
Packit 030a23
{
Packit 030a23
    arm_cpu_features_t features = 0;
Packit 030a23
Packit 030a23
    __try
Packit 030a23
    {
Packit 030a23
	pixman_msvc_try_arm_simd_op ();
Packit 030a23
	features |= ARM_V6;
Packit 030a23
    }
Packit 030a23
    __except (GetExceptionCode () == EXCEPTION_ILLEGAL_INSTRUCTION)
Packit 030a23
    {
Packit 030a23
    }
Packit 030a23
Packit 030a23
    __try
Packit 030a23
    {
Packit 030a23
	pixman_msvc_try_arm_neon_op ();
Packit 030a23
	features |= ARM_NEON;
Packit 030a23
    }
Packit 030a23
    __except (GetExceptionCode () == EXCEPTION_ILLEGAL_INSTRUCTION)
Packit 030a23
    {
Packit 030a23
    }
Packit 030a23
Packit 030a23
    return features;
Packit 030a23
}
Packit 030a23
Packit 030a23
#elif defined(__APPLE__) && defined(TARGET_OS_IPHONE) /* iOS */
Packit 030a23
Packit 030a23
#include "TargetConditionals.h"
Packit 030a23
Packit 030a23
static arm_cpu_features_t
Packit 030a23
detect_cpu_features (void)
Packit 030a23
{
Packit 030a23
    arm_cpu_features_t features = 0;
Packit 030a23
Packit 030a23
    features |= ARM_V6;
Packit 030a23
Packit 030a23
    /* Detection of ARM NEON on iOS is fairly simple because iOS binaries
Packit 030a23
     * contain separate executable images for each processor architecture.
Packit 030a23
     * So all we have to do is detect the armv7 architecture build. The
Packit 030a23
     * operating system automatically runs the armv7 binary for armv7 devices
Packit 030a23
     * and the armv6 binary for armv6 devices.
Packit 030a23
     */
Packit 030a23
#if defined(__ARM_NEON__)
Packit 030a23
    features |= ARM_NEON;
Packit 030a23
#endif
Packit 030a23
Packit 030a23
    return features;
Packit 030a23
}
Packit 030a23
Packit 030a23
#elif defined(__ANDROID__) || defined(ANDROID) /* Android */
Packit 030a23
Packit 030a23
#include <cpu-features.h>
Packit 030a23
Packit 030a23
static arm_cpu_features_t
Packit 030a23
detect_cpu_features (void)
Packit 030a23
{
Packit 030a23
    arm_cpu_features_t features = 0;
Packit 030a23
    AndroidCpuFamily cpu_family;
Packit 030a23
    uint64_t cpu_features;
Packit 030a23
Packit 030a23
    cpu_family = android_getCpuFamily();
Packit 030a23
    cpu_features = android_getCpuFeatures();
Packit 030a23
Packit 030a23
    if (cpu_family == ANDROID_CPU_FAMILY_ARM)
Packit 030a23
    {
Packit 030a23
	if (cpu_features & ANDROID_CPU_ARM_FEATURE_ARMv7)
Packit 030a23
	    features |= ARM_V7;
Packit 030a23
Packit 030a23
	if (cpu_features & ANDROID_CPU_ARM_FEATURE_VFPv3)
Packit 030a23
	    features |= ARM_VFP;
Packit 030a23
Packit 030a23
	if (cpu_features & ANDROID_CPU_ARM_FEATURE_NEON)
Packit 030a23
	    features |= ARM_NEON;
Packit 030a23
    }
Packit 030a23
Packit 030a23
    return features;
Packit 030a23
}
Packit 030a23
Packit 030a23
#elif defined (__linux__) /* linux ELF */
Packit 030a23
Packit 030a23
#include <unistd.h>
Packit 030a23
#include <sys/types.h>
Packit 030a23
#include <sys/stat.h>
Packit 030a23
#include <sys/mman.h>
Packit 030a23
#include <fcntl.h>
Packit 030a23
#include <string.h>
Packit 030a23
#include <elf.h>
Packit 030a23
Packit 030a23
static arm_cpu_features_t
Packit 030a23
detect_cpu_features (void)
Packit 030a23
{
Packit 030a23
    arm_cpu_features_t features = 0;
Packit 030a23
    Elf32_auxv_t aux;
Packit 030a23
    int fd;
Packit 030a23
Packit 030a23
    fd = open ("/proc/self/auxv", O_RDONLY);
Packit 030a23
    if (fd >= 0)
Packit 030a23
    {
Packit 030a23
	while (read (fd, &aux, sizeof(Elf32_auxv_t)) == sizeof(Elf32_auxv_t))
Packit 030a23
	{
Packit 030a23
	    if (aux.a_type == AT_HWCAP)
Packit 030a23
	    {
Packit 030a23
		uint32_t hwcap = aux.a_un.a_val;
Packit 030a23
Packit 030a23
		/* hardcode these values to avoid depending on specific
Packit 030a23
		 * versions of the hwcap header, e.g. HWCAP_NEON
Packit 030a23
		 */
Packit 030a23
		if ((hwcap & 64) != 0)
Packit 030a23
		    features |= ARM_VFP;
Packit 030a23
		if ((hwcap & 512) != 0)
Packit 030a23
		    features |= ARM_IWMMXT;
Packit 030a23
		/* this flag is only present on kernel 2.6.29 */
Packit 030a23
		if ((hwcap & 4096) != 0)
Packit 030a23
		    features |= ARM_NEON;
Packit 030a23
	    }
Packit 030a23
	    else if (aux.a_type == AT_PLATFORM)
Packit 030a23
	    {
Packit 030a23
		const char *plat = (const char*) aux.a_un.a_val;
Packit 030a23
Packit 030a23
		if (strncmp (plat, "v7l", 3) == 0)
Packit 030a23
		    features |= (ARM_V7 | ARM_V6);
Packit 030a23
		else if (strncmp (plat, "v6l", 3) == 0)
Packit 030a23
		    features |= ARM_V6;
Packit 030a23
	    }
Packit 030a23
	}
Packit 030a23
	close (fd);
Packit 030a23
    }
Packit 030a23
Packit 030a23
    return features;
Packit 030a23
}
Packit 030a23
Packit 030a23
#else /* Unknown */
Packit 030a23
Packit 030a23
static arm_cpu_features_t
Packit 030a23
detect_cpu_features (void)
Packit 030a23
{
Packit 030a23
    return 0;
Packit 030a23
}
Packit 030a23
Packit 030a23
#endif /* Linux elf */
Packit 030a23
Packit 030a23
static pixman_bool_t
Packit 030a23
have_feature (arm_cpu_features_t feature)
Packit 030a23
{
Packit 030a23
    static pixman_bool_t initialized;
Packit 030a23
    static arm_cpu_features_t features;
Packit 030a23
Packit 030a23
    if (!initialized)
Packit 030a23
    {
Packit 030a23
	features = detect_cpu_features();
Packit 030a23
	initialized = TRUE;
Packit 030a23
    }
Packit 030a23
Packit 030a23
    return (features & feature) == feature;
Packit 030a23
}
Packit 030a23
Packit 030a23
#endif /* USE_ARM_SIMD || USE_ARM_NEON || USE_ARM_IWMMXT */
Packit 030a23
Packit 030a23
pixman_implementation_t *
Packit 030a23
_pixman_arm_get_implementations (pixman_implementation_t *imp)
Packit 030a23
{
Packit 030a23
#ifdef USE_ARM_SIMD
Packit 030a23
    if (!_pixman_disabled ("arm-simd") && have_feature (ARM_V6))
Packit 030a23
	imp = _pixman_implementation_create_arm_simd (imp);
Packit 030a23
#endif
Packit 030a23
Packit 030a23
#ifdef USE_ARM_IWMMXT
Packit 030a23
    if (!_pixman_disabled ("arm-iwmmxt") && have_feature (ARM_IWMMXT))
Packit 030a23
	imp = _pixman_implementation_create_mmx (imp);
Packit 030a23
#endif
Packit 030a23
Packit 030a23
#ifdef USE_ARM_NEON
Packit 030a23
    if (!_pixman_disabled ("arm-neon") && have_feature (ARM_NEON))
Packit 030a23
	imp = _pixman_implementation_create_arm_neon (imp);
Packit 030a23
#endif
Packit 030a23
Packit 030a23
    return imp;
Packit 030a23
}