Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
Tetragramm
GitHub Repository: Tetragramm/opencv
Path: blob/master/platforms/android/build_sdk.py
16337 views
1
#!/usr/bin/env python
2
3
import os, sys, subprocess, argparse, shutil, glob, re
4
import logging as log
5
import xml.etree.ElementTree as ET
6
7
SCRIPT_DIR = os.path.dirname(os.path.abspath(__file__))
8
9
class Fail(Exception):
10
def __init__(self, text=None):
11
self.t = text
12
def __str__(self):
13
return "ERROR" if self.t is None else self.t
14
15
def execute(cmd, shell=False):
16
try:
17
log.debug("Executing: %s" % cmd)
18
log.info('Executing: ' + ' '.join(cmd))
19
retcode = subprocess.call(cmd, shell=shell)
20
if retcode < 0:
21
raise Fail("Child was terminated by signal: %s" % -retcode)
22
elif retcode > 0:
23
raise Fail("Child returned: %s" % retcode)
24
except OSError as e:
25
raise Fail("Execution failed: %d / %s" % (e.errno, e.strerror))
26
27
def rm_one(d):
28
d = os.path.abspath(d)
29
if os.path.exists(d):
30
if os.path.isdir(d):
31
log.info("Removing dir: %s", d)
32
shutil.rmtree(d)
33
elif os.path.isfile(d):
34
log.info("Removing file: %s", d)
35
os.remove(d)
36
37
def check_dir(d, create=False, clean=False):
38
d = os.path.abspath(d)
39
log.info("Check dir %s (create: %s, clean: %s)", d, create, clean)
40
if os.path.exists(d):
41
if not os.path.isdir(d):
42
raise Fail("Not a directory: %s" % d)
43
if clean:
44
for x in glob.glob(os.path.join(d, "*")):
45
rm_one(x)
46
else:
47
if create:
48
os.makedirs(d)
49
return d
50
51
def determine_engine_version(manifest_path):
52
with open(manifest_path, "rt") as f:
53
return re.search(r'android:versionName="(\d+\.\d+)"', f.read(), re.MULTILINE).group(1)
54
55
def determine_opencv_version(version_hpp_path):
56
# version in 2.4 - CV_VERSION_EPOCH.CV_VERSION_MAJOR.CV_VERSION_MINOR.CV_VERSION_REVISION
57
# version in master - CV_VERSION_MAJOR.CV_VERSION_MINOR.CV_VERSION_REVISION-CV_VERSION_STATUS
58
with open(version_hpp_path, "rt") as f:
59
data = f.read()
60
major = re.search(r'^#define\W+CV_VERSION_MAJOR\W+(\d+)$', data, re.MULTILINE).group(1)
61
minor = re.search(r'^#define\W+CV_VERSION_MINOR\W+(\d+)$', data, re.MULTILINE).group(1)
62
revision = re.search(r'^#define\W+CV_VERSION_REVISION\W+(\d+)$', data, re.MULTILINE).group(1)
63
version_status = re.search(r'^#define\W+CV_VERSION_STATUS\W+"([^"]*)"$', data, re.MULTILINE).group(1)
64
return "%(major)s.%(minor)s.%(revision)s%(version_status)s" % locals()
65
66
# shutil.move fails if dst exists
67
def move_smart(src, dst):
68
def move_recurse(subdir):
69
s = os.path.join(src, subdir)
70
d = os.path.join(dst, subdir)
71
if os.path.exists(d):
72
if os.path.isdir(d):
73
for item in os.listdir(s):
74
move_recurse(os.path.join(subdir, item))
75
elif os.path.isfile(s):
76
shutil.move(s, d)
77
else:
78
shutil.move(s, d)
79
move_recurse('')
80
81
# shutil.copytree fails if dst exists
82
def copytree_smart(src, dst):
83
def copy_recurse(subdir):
84
s = os.path.join(src, subdir)
85
d = os.path.join(dst, subdir)
86
if os.path.exists(d):
87
if os.path.isdir(d):
88
for item in os.listdir(s):
89
copy_recurse(os.path.join(subdir, item))
90
elif os.path.isfile(s):
91
shutil.copy2(s, d)
92
else:
93
if os.path.isdir(s):
94
shutil.copytree(s, d)
95
elif os.path.isfile(s):
96
shutil.copy2(s, d)
97
copy_recurse('')
98
99
#===================================================================================================
100
101
class ABI:
102
def __init__(self, platform_id, name, toolchain, ndk_api_level = None, cmake_vars = dict()):
103
self.platform_id = platform_id # platform code to add to apk version (for cmake)
104
self.name = name # general name (official Android ABI identifier)
105
self.toolchain = toolchain # toolchain identifier (for cmake)
106
self.cmake_vars = dict(
107
ANDROID_STL="gnustl_static",
108
ANDROID_ABI=self.name,
109
ANDROID_PLATFORM_ID=platform_id,
110
)
111
if toolchain is not None:
112
self.cmake_vars['ANDROID_TOOLCHAIN_NAME'] = toolchain
113
else:
114
self.cmake_vars['ANDROID_TOOLCHAIN'] = 'clang'
115
self.cmake_vars['ANDROID_STL'] = 'c++_static'
116
if ndk_api_level:
117
self.cmake_vars['ANDROID_NATIVE_API_LEVEL'] = ndk_api_level
118
self.cmake_vars.update(cmake_vars)
119
def __str__(self):
120
return "%s (%s)" % (self.name, self.toolchain)
121
def haveIPP(self):
122
return self.name == "x86" or self.name == "x86_64"
123
124
#===================================================================================================
125
126
class Builder:
127
def __init__(self, workdir, opencvdir, config):
128
self.workdir = check_dir(workdir, create=True)
129
self.opencvdir = check_dir(opencvdir)
130
self.config = config
131
self.libdest = check_dir(os.path.join(self.workdir, "o4a"), create=True, clean=True)
132
self.resultdest = check_dir(os.path.join(self.workdir, 'OpenCV-android-sdk'), create=True, clean=True)
133
self.docdest = check_dir(os.path.join(self.workdir, 'OpenCV-android-sdk', 'sdk', 'java', 'javadoc'), create=True, clean=True)
134
self.extra_packs = []
135
self.opencv_version = determine_opencv_version(os.path.join(self.opencvdir, "modules", "core", "include", "opencv2", "core", "version.hpp"))
136
self.engine_version = determine_engine_version(os.path.join(self.opencvdir, "platforms", "android", "service", "engine", "AndroidManifest.xml"))
137
self.use_ccache = False if config.no_ccache else True
138
139
def get_toolchain_file(self):
140
if not self.config.force_opencv_toolchain:
141
toolchain = os.path.join(os.environ['ANDROID_NDK'], 'build', 'cmake', 'android.toolchain.cmake')
142
if os.path.exists(toolchain):
143
return toolchain
144
toolchain = os.path.join(SCRIPT_DIR, "android.toolchain.cmake")
145
if os.path.exists(toolchain):
146
return toolchain
147
else:
148
raise Fail("Can't find toolchain")
149
150
def get_engine_apk_dest(self, engdest):
151
return os.path.join(engdest, "platforms", "android", "service", "engine", ".build")
152
153
def add_extra_pack(self, ver, path):
154
if path is None:
155
return
156
self.extra_packs.append((ver, check_dir(path)))
157
158
def clean_library_build_dir(self):
159
for d in ["CMakeCache.txt", "CMakeFiles/", "bin/", "libs/", "lib/", "package/", "install/samples/"]:
160
rm_one(d)
161
162
def build_library(self, abi, do_install):
163
cmd = ["cmake", "-GNinja"]
164
cmake_vars = dict(
165
CMAKE_TOOLCHAIN_FILE=self.get_toolchain_file(),
166
WITH_OPENCL="OFF",
167
WITH_IPP=("ON" if abi.haveIPP() else "OFF"),
168
WITH_TBB="ON",
169
BUILD_EXAMPLES="OFF",
170
BUILD_TESTS="OFF",
171
BUILD_PERF_TESTS="OFF",
172
BUILD_DOCS="OFF",
173
BUILD_ANDROID_EXAMPLES="ON",
174
INSTALL_ANDROID_EXAMPLES="ON",
175
)
176
177
if self.config.extra_modules_path is not None:
178
cmd.append("-DOPENCV_EXTRA_MODULES_PATH='%s'" % self.config.extra_modules_path)
179
180
if self.use_ccache == True:
181
cmd.append("-DNDK_CCACHE=ccache")
182
if do_install:
183
cmd.extend(["-DBUILD_TESTS=ON", "-DINSTALL_TESTS=ON"])
184
185
cmake_vars.update(abi.cmake_vars)
186
cmd += [ "-D%s='%s'" % (k, v) for (k, v) in cmake_vars.items() if v is not None]
187
cmd.append(self.opencvdir)
188
execute(cmd)
189
if do_install:
190
execute(["ninja"])
191
for c in ["libs", "dev", "java", "samples"]:
192
execute(["cmake", "-DCOMPONENT=%s" % c, "-P", "cmake_install.cmake"])
193
else:
194
execute(["ninja", "install/strip"])
195
196
def build_engine(self, abi, engdest):
197
cmd = ["cmake", "-GNinja"]
198
cmake_vars = dict(
199
CMAKE_TOOLCHAIN_FILE=self.get_toolchain_file(),
200
WITH_OPENCL="OFF",
201
WITH_IPP="OFF",
202
BUILD_ANDROID_SERVICE = 'ON'
203
)
204
cmake_vars.update(abi.cmake_vars)
205
cmd += [ "-D%s='%s'" % (k, v) for (k, v) in cmake_vars.items() if v is not None]
206
cmd.append(self.opencvdir)
207
execute(cmd)
208
apkdest = self.get_engine_apk_dest(engdest)
209
assert os.path.exists(apkdest), apkdest
210
# Add extra data
211
apkxmldest = check_dir(os.path.join(apkdest, "res", "xml"), create=True)
212
apklibdest = check_dir(os.path.join(apkdest, "libs", abi.name), create=True)
213
for ver, d in self.extra_packs + [("3.4.3", os.path.join(self.libdest, "lib"))]:
214
r = ET.Element("library", attrib={"version": ver})
215
log.info("Adding libraries from %s", d)
216
217
for f in glob.glob(os.path.join(d, abi.name, "*.so")):
218
log.info("Copy file: %s", f)
219
shutil.copy2(f, apklibdest)
220
if "libnative_camera" in f:
221
continue
222
log.info("Register file: %s", os.path.basename(f))
223
n = ET.SubElement(r, "file", attrib={"name": os.path.basename(f)})
224
225
if len(list(r)) > 0:
226
xmlname = os.path.join(apkxmldest, "config%s.xml" % ver.replace(".", ""))
227
log.info("Generating XML config: %s", xmlname)
228
ET.ElementTree(r).write(xmlname, encoding="utf-8")
229
230
execute(["ninja", "opencv_engine"])
231
execute(["ant", "-f", os.path.join(apkdest, "build.xml"), "debug"],
232
shell=(sys.platform == 'win32'))
233
# TODO: Sign apk
234
235
def build_javadoc(self):
236
classpaths = []
237
for dir, _, files in os.walk(os.environ["ANDROID_SDK"]):
238
for f in files:
239
if f == "android.jar" or f == "annotations.jar":
240
classpaths.append(os.path.join(dir, f))
241
cmd = [
242
"javadoc",
243
"-header", "OpenCV %s" % self.opencv_version,
244
"-nodeprecated",
245
"-footer", '<a href="http://docs.opencv.org">OpenCV %s Documentation</a>' % self.opencv_version,
246
"-public",
247
'-sourcepath', os.path.join(self.resultdest, 'sdk', 'java', 'src'),
248
"-d", self.docdest,
249
"-classpath", ":".join(classpaths),
250
'-subpackages', 'org.opencv',
251
]
252
execute(cmd)
253
254
def gather_results(self, engines):
255
# Copy all files
256
root = os.path.join(self.libdest, "install")
257
for item in os.listdir(root):
258
src = os.path.join(root, item)
259
dst = os.path.join(self.resultdest, item)
260
if os.path.isdir(src):
261
log.info("Copy dir: %s", item)
262
if self.config.force_copy:
263
copytree_smart(src, dst)
264
else:
265
move_smart(src, dst)
266
elif os.path.isfile(src):
267
log.info("Copy file: %s", item)
268
if self.config.force_copy:
269
shutil.copy2(src, dst)
270
else:
271
shutil.move(src, dst)
272
273
# Copy engines for all platforms
274
for abi, engdest in engines:
275
log.info("Copy engine: %s (%s)", abi, engdest)
276
f = os.path.join(self.get_engine_apk_dest(engdest), "bin", "opencv_engine-debug.apk")
277
resname = "OpenCV_%s_Manager_%s_%s.apk" % (self.opencv_version, self.engine_version, abi)
278
dst = os.path.join(self.resultdest, "apk", resname)
279
if self.config.force_copy:
280
shutil.copy2(f, dst)
281
else:
282
shutil.move(f, dst)
283
284
# Clean samples
285
path = os.path.join(self.resultdest, "samples")
286
for item in os.listdir(path):
287
item = os.path.join(path, item)
288
if os.path.isdir(item):
289
for name in ["build.xml", "local.properties", "proguard-project.txt"]:
290
rm_one(os.path.join(item, name))
291
292
293
#===================================================================================================
294
295
if __name__ == "__main__":
296
parser = argparse.ArgumentParser(description='Build OpenCV for Android SDK')
297
parser.add_argument("work_dir", nargs='?', default='.', help="Working directory (and output)")
298
parser.add_argument("opencv_dir", nargs='?', default=os.path.join(SCRIPT_DIR, '../..'), help="Path to OpenCV source dir")
299
parser.add_argument('--config', default='ndk-10.config.py', type=str, help="Package build configuration", )
300
parser.add_argument('--ndk_path', help="Path to Android NDK to use for build")
301
parser.add_argument('--sdk_path', help="Path to Android SDK to use for build")
302
parser.add_argument("--extra_modules_path", help="Path to extra modules to use for build")
303
parser.add_argument('--sign_with', help="Certificate to sign the Manager apk")
304
parser.add_argument('--build_doc', action="store_true", help="Build javadoc")
305
parser.add_argument('--no_ccache', action="store_true", help="Do not use ccache during library build")
306
parser.add_argument('--extra_pack', action='append', help="provide extra OpenCV libraries for Manager apk in form <version>:<path-to-native-libs>, for example '2.4.11:unpacked/sdk/native/libs'")
307
parser.add_argument('--force_copy', action="store_true", help="Do not use file move during library build (useful for debug)")
308
parser.add_argument('--force_opencv_toolchain', action="store_true", help="Do not use toolchain from Android NDK")
309
args = parser.parse_args()
310
311
log.basicConfig(format='%(message)s', level=log.DEBUG)
312
log.debug("Args: %s", args)
313
314
if args.ndk_path is not None:
315
os.environ["ANDROID_NDK"] = args.ndk_path
316
if args.sdk_path is not None:
317
os.environ["ANDROID_SDK"] = args.sdk_path
318
319
if os.path.realpath(args.work_dir) == os.path.realpath(SCRIPT_DIR):
320
raise Fail("Specify workdir (building from script directory is not supported)")
321
if os.path.realpath(args.work_dir) == os.path.realpath(args.opencv_dir):
322
raise Fail("Specify workdir (building from OpenCV source directory is not supported)")
323
324
# Relative paths become invalid in sub-directories
325
if args.opencv_dir is not None and not os.path.isabs(args.opencv_dir):
326
args.opencv_dir = os.path.abspath(args.opencv_dir)
327
if args.extra_modules_path is not None and not os.path.isabs(args.extra_modules_path):
328
args.extra_modules_path = os.path.abspath(args.extra_modules_path)
329
330
cpath = args.config
331
if not os.path.exists(cpath):
332
cpath = os.path.join(SCRIPT_DIR, cpath)
333
if not os.path.exists(cpath):
334
raise Fail('Config "%s" is missing' % args.config)
335
with open(cpath, 'r') as f:
336
cfg = f.read()
337
print("Package configuration:")
338
print('=' * 80)
339
print(cfg.strip())
340
print('=' * 80)
341
342
ABIs = None # make flake8 happy
343
exec(compile(cfg, cpath, 'exec'))
344
345
log.info("Android NDK path: %s", os.environ["ANDROID_NDK"])
346
log.info("Android SDK path: %s", os.environ["ANDROID_SDK"])
347
348
builder = Builder(args.work_dir, args.opencv_dir, args)
349
350
log.info("Detected OpenCV version: %s", builder.opencv_version)
351
log.info("Detected Engine version: %s", builder.engine_version)
352
353
if args.extra_pack:
354
for one in args.extra_pack:
355
i = one.find(":")
356
if i > 0 and i < len(one) - 1:
357
builder.add_extra_pack(one[:i], one[i+1:])
358
else:
359
raise Fail("Bad extra pack provided: %s, should be in form '<version>:<path-to-native-libs>'" % one)
360
361
engines = []
362
for i, abi in enumerate(ABIs):
363
do_install = (i == 0)
364
engdest = check_dir(os.path.join(builder.workdir, "build_service_%s" % abi.name), create=True, clean=True)
365
366
log.info("=====")
367
log.info("===== Building library for %s", abi)
368
log.info("=====")
369
370
os.chdir(builder.libdest)
371
builder.clean_library_build_dir()
372
builder.build_library(abi, do_install)
373
374
log.info("=====")
375
log.info("===== Building engine for %s", abi)
376
log.info("=====")
377
378
os.chdir(engdest)
379
builder.build_engine(abi, engdest)
380
engines.append((abi.name, engdest))
381
382
builder.gather_results(engines)
383
384
if args.build_doc:
385
builder.build_javadoc()
386
387
log.info("=====")
388
log.info("===== Build finished")
389
log.info("=====")
390
log.info("SDK location: %s", builder.resultdest)
391
log.info("Documentation location: %s", builder.docdest)
392
393