summaryrefslogtreecommitdiff
path: root/win32
diff options
context:
space:
mode:
authorMax Kellermann <max@duempel.org>2014-12-09 12:59:03 +0100
committerMax Kellermann <max@duempel.org>2014-12-09 14:10:10 +0100
commitdee49d7488bcf62d32108d0ee7046031d005b0d3 (patch)
tree6ed38479ea23ca3ae30591d0d0c3ec3ec642efad /win32
parent1ecde309e4d8c5285037f9ba55beb44e5677b580 (diff)
win32/build.py: script that cross-compiles to Windows
Diffstat (limited to 'win32')
-rwxr-xr-xwin32/build.py395
1 files changed, 395 insertions, 0 deletions
diff --git a/win32/build.py b/win32/build.py
new file mode 100755
index 000000000..a516ece87
--- /dev/null
+++ b/win32/build.py
@@ -0,0 +1,395 @@
+#!/usr/bin/env python3
+
+import os, os.path
+import sys, shutil, subprocess
+import urllib.request
+import hashlib
+import re
+
+configure_args = sys.argv[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')
+
+host_arch = 'i686-w64-mingw32'
+gcc_toolchain = '/usr'
+
+def select_toolchain():
+ global cc, cxx, ar, nm, strip, cflags, cxxflags, cppflags, ldflags, libs
+
+ target_arch = ''
+ 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')
+ nm = os.path.join(gcc_toolchain, 'bin', host_arch + '-nm')
+ strip = os.path.join(gcc_toolchain, 'bin', host_arch + '-strip')
+
+ cflags = '-O2 -g ' + target_arch
+ cxxflags = '-O2 -g ' + target_arch
+ cppflags = '-I' + root_path + '/include'
+ ldflags = '-L' + root_path + '/lib'
+ libs = ''
+
+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):
+ 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
+
+ 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()
+ 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,
+ '--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 ZlibProject(Project):
+ def __init__(self, url, md5, installed,
+ **kwargs):
+ Project.__init__(self, url, md5, installed, **kwargs)
+
+ def build(self):
+ src = self.unpack()
+
+ build = self.make_build_path()
+
+ select_toolchain()
+ subprocess.check_call(['/usr/bin/make', '--quiet',
+ '-f', 'win32/Makefile.gcc',
+ 'PREFIX=' + host_arch + '-',
+ '-j12',
+ 'install',
+ 'DESTDIR=' + root_path + '/',
+ 'INCLUDE_PATH=include',
+ 'LIBRARY_PATH=lib',
+ 'BINARY_PATH=bin', 'SHARED_MODE=1'],
+ cwd=src)
+
+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()
+ configure = [
+ os.path.join(src, 'configure'),
+ '--cc=' + cc,
+ '--cxx=' + cxx,
+ '--nm=' + nm,
+ '--extra-cflags=' + cflags + ' ' + cppflags + ' ' + self.cppflags,
+ '--extra-cxxflags=' + cxxflags + ' ' + cppflags + ' ' + self.cppflags,
+ '--extra-ldflags=' + ldflags,
+ '--extra-libs=' + libs,
+ '--ar=' + ar,
+ '--enable-cross-compile',
+ '--arch=x86',
+ '--target-os=mingw32',
+ '--cross-prefix=' + host_arch + '-',
+ '--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'],
+ ),
+
+ AutotoolsProject(
+ 'http://downloads.xiph.org/releases/flac/flac-1.3.1.tar.xz',
+ 'b9922c9a0378c88d3e901b234f852698',
+ 'lib/libFLAC.a',
+ [
+ '--disable-shared', '--enable-static',
+ '--disable-xmms-plugin', '--disable-cpplibs',
+ ],
+ ),
+
+ ZlibProject(
+ 'http://zlib.net/zlib-1.2.8.tar.xz',
+ '28f1205d8dd2001f26fec1e8c2cebe37',
+ 'lib/libz.a',
+ ),
+
+ AutotoolsProject(
+ 'ftp://ftp.mars.org/pub/mpeg/libid3tag-0.15.1b.tar.gz',
+ 'e5808ad997ba32c498803822078748c3',
+ 'lib/libid3tag.a',
+ ['--disable-shared', '--enable-static'],
+ autogen=True,
+ ),
+
+ FfmpegProject(
+ 'http://ffmpeg.org/releases/ffmpeg-2.5.tar.bz2',
+ '4346fe710cc6bdd981f6534d2420d1ab',
+ 'lib/libavcodec.a',
+ [
+ '--disable-shared', '--enable-static',
+ '--enable-gpl',
+ '--enable-small',
+ '--disable-pthreads',
+ '--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.39.0.tar.lzma',
+ 'e9aa6dec29920eba8ef706ea5823bad7',
+ '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',
+ ],
+ ),
+
+ 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()
+
+configure = [
+ os.path.join(mpd_path, 'configure'),
+ 'CC=' + cc,
+ 'CXX=' + cxx,
+ 'CFLAGS=' + cflags,
+ 'CXXFLAGS=' + cxxflags,
+ 'CPPFLAGS=' + cppflags,
+ 'LDFLAGS=' + ldflags + ' -static',
+ 'LIBS=' + libs,
+ 'AR=' + ar,
+ 'STRIP=' + strip,
+ '--host=' + host_arch,
+ '--prefix=' + root_path,
+
+ '--enable-silent-rules',
+
+ '--disable-glib',
+ '--disable-icu',
+
+] + configure_args
+
+subprocess.check_call(configure)
+subprocess.check_call(['/usr/bin/make', '--quiet', '-j12'])