summaryrefslogtreecommitdiffstats
path: root/src/pulsecore/cpu-arm.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/pulsecore/cpu-arm.c')
-rw-r--r--src/pulsecore/cpu-arm.c170
1 files changed, 170 insertions, 0 deletions
diff --git a/src/pulsecore/cpu-arm.c b/src/pulsecore/cpu-arm.c
new file mode 100644
index 0000000..122a72b
--- /dev/null
+++ b/src/pulsecore/cpu-arm.c
@@ -0,0 +1,170 @@
+/***
+ This file is part of PulseAudio.
+
+ Copyright 2004-2006 Lennart Poettering
+ Copyright 2009 Wim Taymans <wim.taymans@collabora.co.uk>
+
+ PulseAudio is free software; you can redistribute it and/or modify
+ it under the terms of the GNU Lesser General Public License as published
+ by the Free Software Foundation; either version 2.1 of the License,
+ or (at your option) any later version.
+
+ PulseAudio is distributed in the hope that it will be useful, but
+ WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ General Public License for more details.
+
+ You should have received a copy of the GNU Lesser General Public License
+ along with PulseAudio; if not, see <http://www.gnu.org/licenses/>.
+***/
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include <stdint.h>
+#include <sys/types.h>
+#include <fcntl.h>
+
+#include <pulse/xmalloc.h>
+#include <pulsecore/core-util.h>
+#include <pulsecore/log.h>
+
+#include "cpu-arm.h"
+
+#if defined (__arm__) && defined (__linux__)
+
+#define MAX_BUFFER 4096
+static char *
+get_cpuinfo_line(char *cpuinfo, const char *tag) {
+ char *line, *end, *colon;
+
+ if (!(line = strstr(cpuinfo, tag)))
+ return NULL;
+
+ if (!(end = strchr(line, '\n')))
+ return NULL;
+
+ if (!(colon = strchr(line, ':')))
+ return NULL;
+
+ if (++colon >= end)
+ return NULL;
+
+ return pa_xstrndup(colon, end - colon);
+}
+
+static char *get_cpuinfo(void) {
+ char *cpuinfo;
+ int n, fd;
+
+ cpuinfo = pa_xmalloc(MAX_BUFFER);
+
+ if ((fd = pa_open_cloexec("/proc/cpuinfo", O_RDONLY, 0)) < 0) {
+ pa_xfree(cpuinfo);
+ return NULL;
+ }
+
+ if ((n = pa_read(fd, cpuinfo, MAX_BUFFER-1, NULL)) < 0) {
+ pa_xfree(cpuinfo);
+ pa_close(fd);
+ return NULL;
+ }
+ cpuinfo[n] = 0;
+ pa_close(fd);
+
+ return cpuinfo;
+}
+#endif /* defined (__arm__) && defined (__linux__) */
+
+void pa_cpu_get_arm_flags(pa_cpu_arm_flag_t *flags) {
+#if defined (__arm__) && defined (__linux__)
+ char *cpuinfo, *line;
+ int arch, part;
+
+ /* We need to read the CPU flags from /proc/cpuinfo because there is no user
+ * space support to get the CPU features. This only works on linux AFAIK. */
+ if (!(cpuinfo = get_cpuinfo())) {
+ pa_log("Can't read cpuinfo");
+ return;
+ }
+
+ *flags = 0;
+
+ /* get the CPU architecture */
+ if ((line = get_cpuinfo_line(cpuinfo, "CPU architecture"))) {
+ arch = strtoul(line, NULL, 0);
+ if (arch >= 6)
+ *flags |= PA_CPU_ARM_V6;
+ if (arch >= 7)
+ *flags |= PA_CPU_ARM_V7;
+
+ pa_xfree(line);
+ }
+
+ /* get the CPU features */
+ if ((line = get_cpuinfo_line(cpuinfo, "Features"))) {
+ const char *state = NULL;
+ char *current;
+
+ while ((current = pa_split_spaces(line, &state))) {
+ if (pa_streq(current, "vfp"))
+ *flags |= PA_CPU_ARM_VFP;
+ else if (pa_streq(current, "edsp"))
+ *flags |= PA_CPU_ARM_EDSP;
+ else if (pa_streq(current, "neon"))
+ *flags |= PA_CPU_ARM_NEON;
+ else if (pa_streq(current, "vfpv3"))
+ *flags |= PA_CPU_ARM_VFPV3;
+
+ pa_xfree(current);
+ }
+ pa_xfree(line);
+ }
+
+ /* get the CPU part number */
+ if ((line = get_cpuinfo_line(cpuinfo, "CPU part"))) {
+ part = strtoul(line, NULL, 0);
+ if (part == 0xc08)
+ *flags |= PA_CPU_ARM_CORTEX_A8;
+ pa_xfree(line);
+ }
+ pa_xfree(cpuinfo);
+
+ pa_log_info("CPU flags: %s%s%s%s%s%s%s",
+ (*flags & PA_CPU_ARM_V6) ? "V6 " : "",
+ (*flags & PA_CPU_ARM_V7) ? "V7 " : "",
+ (*flags & PA_CPU_ARM_VFP) ? "VFP " : "",
+ (*flags & PA_CPU_ARM_EDSP) ? "EDSP " : "",
+ (*flags & PA_CPU_ARM_NEON) ? "NEON " : "",
+ (*flags & PA_CPU_ARM_VFPV3) ? "VFPV3 " : "",
+ (*flags & PA_CPU_ARM_CORTEX_A8) ? "Cortex-A8 " : "");
+#endif
+}
+
+bool pa_cpu_init_arm(pa_cpu_arm_flag_t *flags) {
+#if defined (__arm__)
+#if defined (__linux__)
+ pa_cpu_get_arm_flags(flags);
+
+ if (*flags & PA_CPU_ARM_V6)
+ pa_volume_func_init_arm(*flags);
+
+#ifdef HAVE_NEON
+ if (*flags & PA_CPU_ARM_NEON) {
+ pa_convert_func_init_neon(*flags);
+ pa_mix_func_init_neon(*flags);
+ pa_remap_func_init_neon(*flags);
+ }
+#endif
+
+ return true;
+
+#else /* defined (__linux__) */
+ pa_log("Reading ARM CPU features not yet supported on this OS");
+#endif /* defined (__linux__) */
+
+#else /* defined (__arm__) */
+ return false;
+#endif /* defined (__arm__) */
+}