diff options
Diffstat (limited to 'android')
-rw-r--r-- | android/.gitignore | 1 | ||||
-rw-r--r-- | android/AndroidManifest.xml | 24 | ||||
-rwxr-xr-x | android/build.py | 433 | ||||
-rw-r--r-- | android/custom_rules.xml | 11 | ||||
-rw-r--r-- | android/res/values/strings.xml | 5 | ||||
-rw-r--r-- | android/src/Bridge.java | 30 | ||||
-rw-r--r-- | android/src/Loader.java | 39 | ||||
-rw-r--r-- | android/src/Main.java | 75 |
8 files changed, 618 insertions, 0 deletions
diff --git a/android/.gitignore b/android/.gitignore new file mode 100644 index 000000000..796b96d1c --- /dev/null +++ b/android/.gitignore @@ -0,0 +1 @@ +/build diff --git a/android/AndroidManifest.xml b/android/AndroidManifest.xml new file mode 100644 index 000000000..1a65d4f64 --- /dev/null +++ b/android/AndroidManifest.xml @@ -0,0 +1,24 @@ +<?xml version="1.0" encoding="utf-8"?> +<manifest xmlns:android="http://schemas.android.com/apk/res/android" + package="org.musicpd" + android:installLocation="auto" + android:versionCode="5" + android:versionName="0.19~git5"> + + <uses-sdk android:minSdkVersion="9" android:targetSdkVersion="17"/> + + <application android:icon="@drawable/icon" android:label="@string/app_name"> + <activity android:name=".Main" + android:label="@string/app_name" + android:launchMode="singleInstance"> + <intent-filter> + <action android:name="android.intent.action.MAIN" /> + <category android:name="android.intent.category.LAUNCHER" /> + </intent-filter> + </activity> + </application> + + <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/> + <uses-permission android:name="android.permission.WAKE_LOCK"/> + <uses-permission android:name="android.permission.INTERNET"/> +</manifest> diff --git a/android/build.py b/android/build.py new file mode 100755 index 000000000..db475b43d --- /dev/null +++ b/android/build.py @@ -0,0 +1,433 @@ +#!/usr/bin/env python3 + +import os, os.path +import sys, shutil, subprocess +import urllib.request +import hashlib +import re + +if len(sys.argv) < 3: + print("Usage: build.py SDK_PATH NDK_PATH [configure_args...]", file=sys.stderr) + sys.exit(1) + +sdk_path = sys.argv[1] +ndk_path = sys.argv[2] +configure_args = sys.argv[3:] + +if not os.path.isfile(os.path.join(sdk_path, 'tools', 'android')): + print("SDK not found in", ndk_path, file=sys.stderr) + sys.exit(1) + +if not os.path.isdir(ndk_path): + print("NDK not found in", ndk_path, file=sys.stderr) + sys.exit(1) + +# the path to the MPD sources +mpd_path = os.path.dirname(os.path.dirname(sys.argv[0])) + +# output directories +lib_path = os.path.abspath('lib') +tarball_path = lib_path +src_path = os.path.join(lib_path, 'src') +build_path = os.path.join(lib_path, 'build') +root_path = os.path.join(lib_path, 'root') + +# build host configuration +build_arch = 'linux-x86_64' + +# redirect pkg-config to use our root directory instead of the default +# one on the build host +os.environ['PKG_CONFIG_LIBDIR'] = os.path.join(root_path, 'lib/pkgconfig') + +# select the NDK compiler +gcc_version = '4.8' +llvm_version = '3.3' + +# select the NDK target +ndk_arch = 'arm' +host_arch = 'arm-linux-androideabi' +android_abi = 'armeabi-v7a' +ndk_platform = 'android-14' + +# set up the NDK toolchain + +gcc_toolchain = os.path.join(ndk_path, 'toolchains', host_arch + '-' + gcc_version, 'prebuilt', build_arch) +llvm_toolchain = os.path.join(ndk_path, 'toolchains', 'llvm-' + llvm_version, 'prebuilt', build_arch) +ndk_platform_path = os.path.join(ndk_path, 'platforms', ndk_platform) +target_root = os.path.join(ndk_platform_path, 'arch-' + ndk_arch) + +llvm_triple = 'armv7-none-linux-androideabi' + +def select_toolchain(use_cxx, use_clang): + global cc, cxx, ar, strip, cflags, cxxflags, cppflags, ldflags, libs + + target_arch = '-march=armv7-a -mfloat-abi=softfp' + if use_clang: + cc = os.path.join(llvm_toolchain, 'bin/clang') + cxx = os.path.join(llvm_toolchain, 'bin/clang++') + target_arch += ' -target ' + llvm_triple + ' -integrated-as -gcc-toolchain ' + gcc_toolchain + else: + cc = os.path.join(gcc_toolchain, 'bin', host_arch + '-gcc') + cxx = os.path.join(gcc_toolchain, 'bin', host_arch + '-g++') + ar = os.path.join(gcc_toolchain, 'bin', host_arch + '-ar') + strip = os.path.join(gcc_toolchain, 'bin', host_arch + '-strip') + + libstdcxx_path = os.path.join(ndk_path, 'sources/cxx-stl/gnu-libstdc++', gcc_version) + libstdcxx_cppflags = '-isystem ' + os.path.join(libstdcxx_path, 'include') + ' -isystem ' + os.path.join(libstdcxx_path, 'libs', android_abi, 'include') + if use_clang: + libstdcxx_cppflags += ' -D__STRICT_ANSI__' + libstdcxx_ldadd = os.path.join(libstdcxx_path, 'libs', android_abi, 'libgnustl_static.a') + + cflags = '-Os -g ' + target_arch + cxxflags = '-Os -g ' + target_arch + cppflags = '--sysroot=' + target_root + ' -I' + root_path + '/include' + ldflags = '--sysroot=' + target_root + ' -L' + root_path + '/lib' + libs = '' + + if use_cxx: + libs += ' ' + libstdcxx_ldadd + cppflags += ' ' + libstdcxx_cppflags + +def file_md5(path): + """Calculate the MD5 checksum of a file and return it in hexadecimal notation.""" + + with open(path, 'rb') as f: + m = hashlib.md5() + while True: + data = f.read(65536) + if len(data) == 0: + # end of file + return m.hexdigest() + m.update(data) + +def download_tarball(url, md5): + """Download a tarball, verify its MD5 checksum and return the local path.""" + + global tarball_path + os.makedirs(tarball_path, exist_ok=True) + path = os.path.join(tarball_path, os.path.basename(url)) + + try: + calculated_md5 = file_md5(path) + if md5 == calculated_md5: return path + os.unlink(path) + except FileNotFoundError: + pass + + tmp_path = path + '.tmp' + + print("download", url) + urllib.request.urlretrieve(url, tmp_path) + calculated_md5 = file_md5(tmp_path) + if calculated_md5 != md5: + os.unlink(tmp_path) + raise "MD5 mismatch" + + os.rename(tmp_path, path) + return path + +class Project: + def __init__(self, url, md5, installed, name=None, version=None, + base=None, + use_cxx=False, use_clang=False): + if base is None: + basename = os.path.basename(url) + m = re.match(r'^(.+)\.(tar(\.(gz|bz2|xz|lzma))?|zip)$', basename) + if not m: raise + self.base = m.group(1) + else: + self.base = base + + if name is None or version is None: + m = re.match(r'^([-\w]+)-(\d[\d.]*[a-z]?)$', self.base) + if name is None: name = m.group(1) + if version is None: version = m.group(2) + + self.name = name + self.version = version + + self.url = url + self.md5 = md5 + self.installed = installed + + self.use_cxx = use_cxx + self.use_clang = use_clang + + def download(self): + return download_tarball(self.url, self.md5) + + def is_installed(self): + global root_path + tarball = self.download() + installed = os.path.join(root_path, self.installed) + tarball_mtime = os.path.getmtime(tarball) + try: + return os.path.getmtime(installed) >= tarball_mtime + except FileNotFoundError: + return False + + def unpack(self): + global src_path + tarball = self.download() + path = os.path.join(src_path, self.base) + try: + shutil.rmtree(path) + except FileNotFoundError: + pass + os.makedirs(src_path, exist_ok=True) + subprocess.check_call(['/bin/tar', 'xfC', tarball, src_path]) + return path + + def make_build_path(self): + path = os.path.join(build_path, self.base) + try: + shutil.rmtree(path) + except FileNotFoundError: + pass + os.makedirs(path, exist_ok=True) + return path + +class AutotoolsProject(Project): + def __init__(self, url, md5, installed, configure_args=[], + autogen=False, + cppflags='', + **kwargs): + Project.__init__(self, url, md5, installed, **kwargs) + self.configure_args = configure_args + self.autogen = autogen + self.cppflags = cppflags + + def build(self): + src = self.unpack() + if self.autogen: + subprocess.check_call(['/usr/bin/aclocal'], cwd=src) + subprocess.check_call(['/usr/bin/automake', '--add-missing', '--force-missing', '--foreign'], cwd=src) + subprocess.check_call(['/usr/bin/autoconf'], cwd=src) + subprocess.check_call(['/usr/bin/libtoolize', '--force'], cwd=src) + + build = self.make_build_path() + + select_toolchain(use_cxx=self.use_cxx, use_clang=self.use_clang) + configure = [ + os.path.join(src, 'configure'), + 'CC=' + cc, + 'CXX=' + cxx, + 'CFLAGS=' + cflags, + 'CXXFLAGS=' + cxxflags, + 'CPPFLAGS=' + cppflags + ' ' + self.cppflags, + 'LDFLAGS=' + ldflags, + 'LIBS=' + libs, + 'AR=' + ar, + 'STRIP=' + strip, + '--host=' + host_arch, + '--prefix=' + root_path, + '--with-sysroot=' + target_root, + '--enable-silent-rules', + ] + self.configure_args + + subprocess.check_call(configure, cwd=build) + subprocess.check_call(['/usr/bin/make', '--quiet', '-j12'], cwd=build) + subprocess.check_call(['/usr/bin/make', '--quiet', 'install'], cwd=build) + +class FfmpegProject(Project): + def __init__(self, url, md5, installed, configure_args=[], + cppflags='', + **kwargs): + Project.__init__(self, url, md5, installed, **kwargs) + self.configure_args = configure_args + self.cppflags = cppflags + + def build(self): + src = self.unpack() + build = self.make_build_path() + + select_toolchain(use_cxx=self.use_cxx, use_clang=self.use_clang) + configure = [ + os.path.join(src, 'configure'), + '--cc=' + cc, + '--cxx=' + cxx, + '--extra-cflags=' + cflags + ' ' + cppflags + ' ' + self.cppflags, + '--extra-cxxflags=' + cxxflags + ' ' + cppflags + ' ' + self.cppflags, + '--extra-ldflags=' + ldflags, + '--extra-libs=' + libs, + '--ar=' + ar, + '--enable-cross-compile', + '--target-os=linux', + '--arch=' + ndk_arch, + '--cpu=cortex-a8', + '--prefix=' + root_path, + ] + self.configure_args + + subprocess.check_call(configure, cwd=build) + subprocess.check_call(['/usr/bin/make', '--quiet', '-j12'], cwd=build) + subprocess.check_call(['/usr/bin/make', '--quiet', 'install'], cwd=build) + +class BoostProject(Project): + def __init__(self, url, md5, installed, + **kwargs): + m = re.match(r'.*/boost_(\d+)_(\d+)_(\d+)\.tar\.bz2$', url) + version = "%s.%s.%s" % (m.group(1), m.group(2), m.group(3)) + Project.__init__(self, url, md5, installed, + name='boost', version=version, + **kwargs) + + def build(self): + src = self.unpack() + + # install the headers manually; don't build any library + # (because right now, we only use header-only libraries) + includedir = os.path.join(root_path, 'include') + for dirpath, dirnames, filenames in os.walk(os.path.join(src, 'boost')): + relpath = dirpath[len(src)+1:] + destdir = os.path.join(includedir, relpath) + try: + os.mkdir(destdir) + except: + pass + for name in filenames: + if name[-4:] == '.hpp': + shutil.copyfile(os.path.join(dirpath, name), + os.path.join(destdir, name)) + +# a list of third-party libraries to be used by MPD on Android +thirdparty_libs = [ + AutotoolsProject( + 'http://downloads.xiph.org/releases/ogg/libogg-1.3.2.tar.xz', + '5c3a34309d8b98640827e5d0991a4015', + 'lib/libogg.a', + ['--disable-shared', '--enable-static'], + ), + + AutotoolsProject( + 'http://downloads.xiph.org/releases/vorbis/libvorbis-1.3.4.tar.xz', + '55f2288055e44754275a17c9a2497391', + 'lib/libvorbis.a', + ['--disable-shared', '--enable-static'], + ), + + AutotoolsProject( + 'http://downloads.xiph.org/releases/opus/opus-1.1.tar.gz', + 'c5a8cf7c0b066759542bc4ca46817ac6', + 'lib/libopus.a', + ['--disable-shared', '--enable-static'], + use_clang=True, + ), + + AutotoolsProject( + 'https://svn.xiph.org/releases/flac/flac-1.3.0.tar.xz', + '13b5c214cee8373464d3d65dee362cdd', + 'lib/libFLAC.a', + [ + '--disable-shared', '--enable-static', + '--disable-xmms-plugin', '--disable-cpplibs', + ], + use_clang=True, + ), + + AutotoolsProject( + 'ftp://ftp.mars.org/pub/mpeg/libid3tag-0.15.1b.tar.gz', + 'e5808ad997ba32c498803822078748c3', + 'lib/libid3tag.a', + ['--disable-shared', '--enable-static'], + autogen=True, + ), + + AutotoolsProject( + 'ftp://ftp.mars.org/pub/mpeg/libmad-0.15.1b.tar.gz', + '1be543bc30c56fb6bea1d7bf6a64e66c', + 'lib/libmad.a', + ['--disable-shared', '--enable-static'], + autogen=True, + ), + + FfmpegProject( + 'http://www.ffmpeg.org/releases/ffmpeg-2.2.3.tar.bz2', + 'dbb5b6b69bd010916f17df0ae596e0b1', + 'lib/libavcodec.a', + [ + '--disable-shared', '--enable-static', + '--enable-gpl', + '--enable-small', + '--disable-pthreads', + '--disable-runtime-cpudetect', + '--disable-programs', + '--disable-doc', + '--disable-avdevice', + '--disable-swresample', + '--disable-swscale', + '--disable-postproc', + '--disable-avfilter', + '--disable-network', + '--disable-encoders', + '--disable-protocols', + '--disable-outdevs', + '--disable-filters', + ], + ), + + AutotoolsProject( + 'http://curl.haxx.se/download/curl-7.37.0.tar.lzma', + '54bfd1eb5214f604186d6f5ac61c7781', + 'lib/libcurl.a', + [ + '--disable-shared', '--enable-static', + '--disable-debug', + '--enable-http', + '--enable-ipv6', + '--disable-ftp', '--disable-file', + '--disable-ldap', '--disable-ldaps', + '--disable-rtsp', '--disable-proxy', '--disable-dict', '--disable-telnet', + '--disable-tftp', '--disable-pop3', '--disable-imap', '--disable-smtp', + '--disable-gopher', + '--disable-manual', + '--disable-threaded-resolver', '--disable-verbose', '--disable-sspi', + '--disable-crypto-auth', '--disable-ntlm-wb', '--disable-tls-srp', '--disable-cookies', + '--without-ssl', '--without-gnutls', '--without-nss', '--without-libssh2', + ], + use_clang=True, + ), + + BoostProject( + 'http://netcologne.dl.sourceforge.net/project/boost/boost/1.55.0/boost_1_55_0.tar.bz2', + 'd6eef4b4cacb2183f2bf265a5a03a354', + 'include/boost/version.hpp', + ), +] + +# build the third-party libraries +for x in thirdparty_libs: + if not x.is_installed(): + x.build() + +# configure and build MPD +select_toolchain(use_cxx=True, use_clang=True) + +configure = [ + os.path.join(mpd_path, 'configure'), + 'CC=' + cc, + 'CXX=' + cxx, + 'CFLAGS=' + cflags, + 'CXXFLAGS=' + cxxflags, + 'CPPFLAGS=' + cppflags, + 'LDFLAGS=' + ldflags, + 'LIBS=' + libs, + 'AR=' + ar, + 'STRIP=' + strip, + '--host=' + host_arch, + '--prefix=' + root_path, + '--with-sysroot=' + target_root, + '--with-android-sdk=' + sdk_path, + + '--enable-silent-rules', + + '--disable-glib', + '--disable-icu', + + # disabled for now because these features require GLib: + '--disable-httpd-output', + '--disable-vorbis-encoder', + +] + configure_args + +subprocess.check_call(configure) +subprocess.check_call(['/usr/bin/make', '--quiet', '-j12']) diff --git a/android/custom_rules.xml b/android/custom_rules.xml new file mode 100644 index 000000000..68db46590 --- /dev/null +++ b/android/custom_rules.xml @@ -0,0 +1,11 @@ +<?xml version="1.0" encoding="UTF-8"?> + +<project name="mpd_rules"> + <!-- setting these two properties works around a bug in Android + SDK's build.xml, which deletes all .class files every time --> + <property name="build.last.is.packaging.debug" value="true" /> + <property name="build.is.packaging.debug" value="true" /> + + <target name="compile-jni-classes" + depends="-set-debug-mode,-compile"/> +</project> diff --git a/android/res/values/strings.xml b/android/res/values/strings.xml new file mode 100644 index 000000000..416c8de9f --- /dev/null +++ b/android/res/values/strings.xml @@ -0,0 +1,5 @@ +<?xml version="1.0" encoding="utf-8"?> + +<resources> + <string name="app_name">MPD</string> +</resources> diff --git a/android/src/Bridge.java b/android/src/Bridge.java new file mode 100644 index 000000000..537343b56 --- /dev/null +++ b/android/src/Bridge.java @@ -0,0 +1,30 @@ +/* + * Copyright (C) 2003-2014 The Music Player Daemon Project + * http://www.musicpd.org + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program 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 General Public License along + * with this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + */ + +package org.musicpd; + +import android.content.Context; + +/** + * Bridge to native code. + */ +public class Bridge { + public static native void run(Context context); + public static native void shutdown(); +} diff --git a/android/src/Loader.java b/android/src/Loader.java new file mode 100644 index 000000000..fdaa27753 --- /dev/null +++ b/android/src/Loader.java @@ -0,0 +1,39 @@ +/* + * Copyright (C) 2003-2014 The Music Player Daemon Project + * http://www.musicpd.org + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program 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 General Public License along + * with this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + */ + +package org.musicpd; + +import android.util.Log; + +public class Loader { + private static final String TAG = "MPD"; + + public static boolean loaded = false; + public static String error; + + static { + try { + System.loadLibrary("mpd"); + loaded = true; + } catch (UnsatisfiedLinkError e) { + Log.e(TAG, e.getMessage()); + error = e.getMessage(); + } + } +} diff --git a/android/src/Main.java b/android/src/Main.java new file mode 100644 index 000000000..d87f7709b --- /dev/null +++ b/android/src/Main.java @@ -0,0 +1,75 @@ +/* + * Copyright (C) 2003-2014 The Music Player Daemon Project + * http://www.musicpd.org + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program 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 General Public License along + * with this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + */ + +package org.musicpd; + +import android.app.Activity; +import android.os.Bundle; +import android.os.Build; +import android.os.Handler; +import android.os.Message; +import android.widget.TextView; +import android.util.Log; + +public class Main extends Activity implements Runnable { + private static final String TAG = "MPD"; + + Thread thread; + + TextView textView; + + final Handler quitHandler = new Handler() { + public void handleMessage(Message msg) { + textView.setText("Music Player Daemon has quit"); + + // TODO: what now? restart? + } + }; + + @Override protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + + if (!Loader.loaded) { + TextView tv = new TextView(this); + tv.setText("Failed to load the native MPD libary.\n" + + "Report this problem to us, and include the following information:\n" + + "ABI=" + Build.CPU_ABI + "\n" + + "PRODUCT=" + Build.PRODUCT + "\n" + + "FINGERPRINT=" + Build.FINGERPRINT + "\n" + + "error=" + Loader.error); + setContentView(tv); + return; + } + + if (thread == null || !thread.isAlive()) { + thread = new Thread(this, "NativeMain"); + thread.start(); + } + + textView = new TextView(this); + textView.setText("Music Player Daemon is running" + + "\nCAUTION: this version is EXPERIMENTAL!"); + setContentView(textView); + } + + @Override public void run() { + Bridge.run(this); + quitHandler.sendMessage(quitHandler.obtainMessage()); + } +} |