blob: 021fd5af814d41428e986d55ac9c97c250449ab5 [file] [log] [blame]
Roland Levillain29e740f2019-08-20 15:31:37 +01001#!/usr/bin/env python
2# -*- coding: utf-8 -*-
Andreas Gampe2c846cd2019-01-29 22:06:46 +00003
4# Copyright (C) 2019 The Android Open Source Project
5#
6# Licensed under the Apache License, Version 2.0 (the "License");
7# you may not use this file except in compliance with the License.
8# You may obtain a copy of the License at
9#
10# http://www.apache.org/licenses/LICENSE-2.0
11#
12# Unless required by applicable law or agreed to in writing, software
13# distributed under the License is distributed on an "AS IS" BASIS,
14# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15# See the License for the specific language governing permissions and
16# limitations under the License.
17#
18
19import argparse
Martin Stjernholm84a32cd2019-03-29 16:39:46 +000020import fnmatch
Andreas Gampe2c846cd2019-01-29 22:06:46 +000021import logging
22import os
Martin Stjernholm84a32cd2019-03-29 16:39:46 +000023import os.path
Andreas Gampe2c846cd2019-01-29 22:06:46 +000024import subprocess
25import sys
26import zipfile
27
28logging.basicConfig(format='%(message)s')
29
Orion Hodsonaf3ed622019-10-03 16:30:54 +010030# Flavors of ART APEX package.
31FLAVOR_RELEASE = 'release'
32FLAVOR_DEBUG = 'debug'
33FLAVOR_TESTING = 'testing'
34FLAVOR_AUTO = 'auto'
35FLAVORS_ALL = [FLAVOR_RELEASE, FLAVOR_DEBUG, FLAVOR_TESTING, FLAVOR_AUTO]
36
37# Bitness options for APEX package
38BITNESS_32 = '32'
39BITNESS_64 = '64'
40BITNESS_MULTILIB = 'multilib'
41BITNESS_AUTO = 'auto'
42BITNESS_ALL = [BITNESS_32, BITNESS_64, BITNESS_MULTILIB, BITNESS_AUTO]
43
Roland Levillain61f07162019-06-26 12:44:04 +010044# Architectures supported by APEX packages.
Orion Hodsonaf3ed622019-10-03 16:30:54 +010045ARCHS = ["arm", "arm64", "x86", "x86_64"]
46
Martin Stjernholme58624f2019-09-20 15:53:40 +010047# Directory containing ART tests within an ART APEX (if the package includes
Roland Levillain61f07162019-06-26 12:44:04 +010048# any). ART test executables are installed in `bin/art/<arch>`. Segregating
49# tests by architecture is useful on devices supporting more than one
Martin Stjernholme58624f2019-09-20 15:53:40 +010050# architecture, as it permits testing all of them using a single ART APEX
Roland Levillain61f07162019-06-26 12:44:04 +010051# package.
Orion Hodsonaf3ed622019-10-03 16:30:54 +010052ART_TEST_DIR = 'bin/art'
Luca Stefani4e91ee92019-03-06 15:08:16 +010053
Ulya Trafimovich7b0df592020-01-03 13:29:00 +000054
55# Test if a given variable is set to a string "true".
56def isEnvTrue(var):
57 return var in os.environ and os.environ[var] == 'true'
58
59
Andreas Gampe2c846cd2019-01-29 22:06:46 +000060class FSObject:
Roland Levillain88e55692019-07-25 15:57:06 +010061 def __init__(self, name, is_dir, is_exec, is_symlink, size):
Andreas Gampe2c846cd2019-01-29 22:06:46 +000062 self.name = name
63 self.is_dir = is_dir
64 self.is_exec = is_exec
65 self.is_symlink = is_symlink
Roland Levillain88e55692019-07-25 15:57:06 +010066 self.size = size
Luca Stefani4e91ee92019-03-06 15:08:16 +010067
Andreas Gampe2c846cd2019-01-29 22:06:46 +000068 def __str__(self):
Roland Levillain88e55692019-07-25 15:57:06 +010069 return '%s(dir=%r,exec=%r,symlink=%r,size=%d)' \
70 % (self.name, self.is_dir, self.is_exec, self.is_symlink, self.size)
Andreas Gampe2c846cd2019-01-29 22:06:46 +000071
Luca Stefani4e91ee92019-03-06 15:08:16 +010072
Andreas Gampe2c846cd2019-01-29 22:06:46 +000073class TargetApexProvider:
74 def __init__(self, apex, tmpdir, debugfs):
75 self._tmpdir = tmpdir
76 self._debugfs = debugfs
77 self._folder_cache = {}
78 self._payload = os.path.join(self._tmpdir, 'apex_payload.img')
79 # Extract payload to tmpdir.
Luca Stefani4e91ee92019-03-06 15:08:16 +010080 apex_zip = zipfile.ZipFile(apex)
81 apex_zip.extract('apex_payload.img', tmpdir)
Andreas Gampe2c846cd2019-01-29 22:06:46 +000082
83 def __del__(self):
84 # Delete temps.
85 if os.path.exists(self._payload):
86 os.remove(self._payload)
87
88 def get(self, path):
Luca Stefani4e91ee92019-03-06 15:08:16 +010089 apex_dir, name = os.path.split(path)
90 if not apex_dir:
91 apex_dir = '.'
92 apex_map = self.read_dir(apex_dir)
93 return apex_map[name] if name in apex_map else None
Andreas Gampe2c846cd2019-01-29 22:06:46 +000094
Luca Stefani4e91ee92019-03-06 15:08:16 +010095 def read_dir(self, apex_dir):
96 if apex_dir in self._folder_cache:
97 return self._folder_cache[apex_dir]
Andreas Gampe2c846cd2019-01-29 22:06:46 +000098 # Cannot use check_output as it will annoy with stderr.
Luca Stefani4e91ee92019-03-06 15:08:16 +010099 process = subprocess.Popen([self._debugfs, '-R', 'ls -l -p %s' % apex_dir, self._payload],
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000100 stdout=subprocess.PIPE, stderr=subprocess.PIPE,
101 universal_newlines=True)
Luca Stefani4e91ee92019-03-06 15:08:16 +0100102 stdout, _ = process.communicate()
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000103 res = str(stdout)
Luca Stefani4e91ee92019-03-06 15:08:16 +0100104 apex_map = {}
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000105 # Debugfs output looks like this:
106 # debugfs 1.44.4 (18-Aug-2018)
107 # /12/040755/0/2000/.//
108 # /2/040755/1000/1000/..//
109 # /13/100755/0/2000/dalvikvm32/28456/
110 # /14/100755/0/2000/dexoptanalyzer/20396/
111 # /15/100755/0/2000/linker/1152724/
112 # /16/100755/0/2000/dex2oat/563508/
113 # /17/100755/0/2000/linker64/1605424/
114 # /18/100755/0/2000/profman/85304/
115 # /19/100755/0/2000/dalvikvm64/28576/
116 # | | | | | |
117 # | | | #- gid #- name #- size
118 # | | #- uid
119 # | #- type and permission bits
120 # #- inode nr (?)
121 #
122 # Note: could break just on '/' to avoid names with newlines.
123 for line in res.split("\n"):
124 if not line:
125 continue
126 comps = line.split('/')
127 if len(comps) != 8:
Luca Stefani4e91ee92019-03-06 15:08:16 +0100128 logging.warning('Could not break and parse line \'%s\'', line)
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000129 continue
130 bits = comps[2]
131 name = comps[5]
Roland Levillain88e55692019-07-25 15:57:06 +0100132 size_str = comps[6]
133 # Use a negative value as an indicator of undefined/unknown size.
134 size = int(size_str) if size_str != '' else -1
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000135 if len(bits) != 6:
Luca Stefani4e91ee92019-03-06 15:08:16 +0100136 logging.warning('Dont understand bits \'%s\'', bits)
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000137 continue
Luca Stefani4e91ee92019-03-06 15:08:16 +0100138 is_dir = bits[1] == '4'
139
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000140 def is_exec_bit(ch):
Luca Stefani4e91ee92019-03-06 15:08:16 +0100141 return int(ch) & 1 == 1
142
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000143 is_exec = is_exec_bit(bits[3]) and is_exec_bit(bits[4]) and is_exec_bit(bits[5])
Luca Stefani4e91ee92019-03-06 15:08:16 +0100144 is_symlink = bits[1] == '2'
Roland Levillain88e55692019-07-25 15:57:06 +0100145 apex_map[name] = FSObject(name, is_dir, is_exec, is_symlink, size)
Luca Stefani4e91ee92019-03-06 15:08:16 +0100146 self._folder_cache[apex_dir] = apex_map
147 return apex_map
148
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000149
Roland Levillaindd20d002019-07-19 16:09:47 +0100150class TargetFlattenedApexProvider:
151 def __init__(self, apex):
152 self._folder_cache = {}
153 self._apex = apex
154
155 def get(self, path):
156 apex_dir, name = os.path.split(path)
157 if not apex_dir:
158 apex_dir = '.'
159 apex_map = self.read_dir(apex_dir)
160 return apex_map[name] if name in apex_map else None
161
162 def read_dir(self, apex_dir):
163 if apex_dir in self._folder_cache:
164 return self._folder_cache[apex_dir]
165 apex_map = {}
166 dirname = os.path.join(self._apex, apex_dir)
167 if os.path.exists(dirname):
168 for basename in os.listdir(dirname):
169 filepath = os.path.join(dirname, basename)
170 is_dir = os.path.isdir(filepath)
171 is_exec = os.access(filepath, os.X_OK)
172 is_symlink = os.path.islink(filepath)
Roland Levillain3cd802e2020-02-20 15:31:20 +0000173 if is_symlink:
174 # Report the length of the symlink's target's path as file size, like `ls`.
175 size = len(os.readlink(filepath))
176 else:
177 size = os.path.getsize(filepath)
Roland Levillain88e55692019-07-25 15:57:06 +0100178 apex_map[basename] = FSObject(basename, is_dir, is_exec, is_symlink, size)
Roland Levillaindd20d002019-07-19 16:09:47 +0100179 self._folder_cache[apex_dir] = apex_map
180 return apex_map
181
182
Andreas Gampe09123952019-01-30 13:17:02 -0800183class HostApexProvider:
184 def __init__(self, apex, tmpdir):
185 self._tmpdir = tmpdir
Roland Levillain77251f92019-08-09 15:00:04 +0100186 self._folder_cache = {}
Andreas Gampe09123952019-01-30 13:17:02 -0800187 self._payload = os.path.join(self._tmpdir, 'apex_payload.zip')
188 # Extract payload to tmpdir.
Luca Stefani4e91ee92019-03-06 15:08:16 +0100189 apex_zip = zipfile.ZipFile(apex)
190 apex_zip.extract('apex_payload.zip', tmpdir)
Andreas Gampe09123952019-01-30 13:17:02 -0800191
192 def __del__(self):
193 # Delete temps.
194 if os.path.exists(self._payload):
195 os.remove(self._payload)
196
197 def get(self, path):
Luca Stefani4e91ee92019-03-06 15:08:16 +0100198 apex_dir, name = os.path.split(path)
199 if not apex_dir:
200 apex_dir = ''
201 apex_map = self.read_dir(apex_dir)
202 return apex_map[name] if name in apex_map else None
Andreas Gampe09123952019-01-30 13:17:02 -0800203
Luca Stefani4e91ee92019-03-06 15:08:16 +0100204 def read_dir(self, apex_dir):
Roland Levillain77251f92019-08-09 15:00:04 +0100205 if apex_dir in self._folder_cache:
206 return self._folder_cache[apex_dir]
207 if not self._folder_cache:
Andreas Gampe09123952019-01-30 13:17:02 -0800208 self.parse_zip()
Roland Levillain77251f92019-08-09 15:00:04 +0100209 if apex_dir in self._folder_cache:
210 return self._folder_cache[apex_dir]
Andreas Gampe09123952019-01-30 13:17:02 -0800211 return {}
212
213 def parse_zip(self):
Luca Stefani4e91ee92019-03-06 15:08:16 +0100214 apex_zip = zipfile.ZipFile(self._payload)
215 infos = apex_zip.infolist()
Andreas Gampe09123952019-01-30 13:17:02 -0800216 for zipinfo in infos:
217 path = zipinfo.filename
218
219 # Assume no empty file is stored.
220 assert path
221
222 def get_octal(val, index):
Luca Stefani4e91ee92019-03-06 15:08:16 +0100223 return (val >> (index * 3)) & 0x7
224
Andreas Gampe09123952019-01-30 13:17:02 -0800225 def bits_is_exec(val):
226 # TODO: Enforce group/other, too?
227 return get_octal(val, 2) & 1 == 1
228
229 is_zipinfo = True
230 while path:
Luca Stefani4e91ee92019-03-06 15:08:16 +0100231 apex_dir, base = os.path.split(path)
Andreas Gampe09123952019-01-30 13:17:02 -0800232 # TODO: If directories are stored, base will be empty.
233
Roland Levillain77251f92019-08-09 15:00:04 +0100234 if apex_dir not in self._folder_cache:
235 self._folder_cache[apex_dir] = {}
236 dir_map = self._folder_cache[apex_dir]
Luca Stefani4e91ee92019-03-06 15:08:16 +0100237 if base not in dir_map:
Andreas Gampe09123952019-01-30 13:17:02 -0800238 if is_zipinfo:
239 bits = (zipinfo.external_attr >> 16) & 0xFFFF
240 is_dir = get_octal(bits, 4) == 4
241 is_symlink = get_octal(bits, 4) == 2
242 is_exec = bits_is_exec(bits)
Roland Levillain88e55692019-07-25 15:57:06 +0100243 size = zipinfo.file_size
Andreas Gampe09123952019-01-30 13:17:02 -0800244 else:
245 is_exec = False # Seems we can't get this easily?
246 is_symlink = False
247 is_dir = True
Roland Levillain88e55692019-07-25 15:57:06 +0100248 # Use a negative value as an indicator of undefined/unknown size.
249 size = -1
250 dir_map[base] = FSObject(base, is_dir, is_exec, is_symlink, size)
Andreas Gampe09123952019-01-30 13:17:02 -0800251 is_zipinfo = False
Luca Stefani4e91ee92019-03-06 15:08:16 +0100252 path = apex_dir
253
Andreas Gampe09123952019-01-30 13:17:02 -0800254
Andreas Gampeeb555b82019-01-30 14:47:49 -0800255# DO NOT USE DIRECTLY! This is an "abstract" base class.
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000256class Checker:
257 def __init__(self, provider):
258 self._provider = provider
259 self._errors = 0
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000260 self._expected_file_globs = set()
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000261
Luca Stefani4e91ee92019-03-06 15:08:16 +0100262 def fail(self, msg, *fail_args):
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000263 self._errors += 1
Martin Stjernholm2babede2019-03-18 21:04:49 +0000264 logging.error(msg, *fail_args)
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000265
266 def error_count(self):
267 return self._errors
Luca Stefani4e91ee92019-03-06 15:08:16 +0100268
Andreas Gampe9dc4b052019-01-30 13:47:25 -0800269 def reset_errors(self):
270 self._errors = 0
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000271
Martin Stjernholm2babede2019-03-18 21:04:49 +0000272 def is_file(self, path):
273 fs_object = self._provider.get(path)
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000274 if fs_object is None:
Luca Stefani4e91ee92019-03-06 15:08:16 +0100275 return False, 'Could not find %s'
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000276 if fs_object.is_dir:
Luca Stefani4e91ee92019-03-06 15:08:16 +0100277 return False, '%s is a directory'
278 return True, ''
Andreas Gampea0242cf2019-01-29 13:01:23 -0800279
Ulyana Trafimovich7a20cbd2019-11-08 11:27:31 +0000280 def is_dir(self, path):
281 fs_object = self._provider.get(path)
282 if fs_object is None:
283 return False, 'Could not find %s'
284 if not fs_object.is_dir:
285 return False, '%s is not a directory'
286 return True, ''
287
Martin Stjernholm2babede2019-03-18 21:04:49 +0000288 def check_file(self, path):
289 ok, msg = self.is_file(path)
290 if not ok:
291 self.fail(msg, path)
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000292 self._expected_file_globs.add(path)
Martin Stjernholm2babede2019-03-18 21:04:49 +0000293 return ok
Luca Stefani4e91ee92019-03-06 15:08:16 +0100294
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000295 def check_executable(self, filename):
Martin Stjernholm2babede2019-03-18 21:04:49 +0000296 path = 'bin/%s' % filename
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000297 if not self.check_file(path):
Luca Stefani4e91ee92019-03-06 15:08:16 +0100298 return
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000299 if not self._provider.get(path).is_exec:
300 self.fail('%s is not executable', path)
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000301
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000302 def check_executable_symlink(self, filename):
Martin Stjernholm2babede2019-03-18 21:04:49 +0000303 path = 'bin/%s' % filename
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000304 fs_object = self._provider.get(path)
305 if fs_object is None:
306 self.fail('Could not find %s', path)
Luca Stefani4e91ee92019-03-06 15:08:16 +0100307 return
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000308 if fs_object.is_dir:
309 self.fail('%s is a directory', path)
Luca Stefani4e91ee92019-03-06 15:08:16 +0100310 return
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000311 if not fs_object.is_symlink:
312 self.fail('%s is not a symlink', path)
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000313 self._expected_file_globs.add(path)
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000314
Ulyana Trafimovich7a20cbd2019-11-08 11:27:31 +0000315 def arch_dirs_for_path(self, path):
316 # Look for target-specific subdirectories for the given directory path.
317 # This is needed because the list of build targets is not propagated
318 # to this script.
Ulyana Trafimovich0f7c7922019-11-07 16:12:09 +0000319 #
Roland Levillain48136182019-11-15 13:39:03 +0000320 # TODO(b/123602136): Pass build target information to this script and fix
321 # all places where this function in used (or similar workarounds).
Ulyana Trafimovich7a20cbd2019-11-08 11:27:31 +0000322 dirs = []
Ulyana Trafimovich0f7c7922019-11-07 16:12:09 +0000323 for arch in ARCHS:
Ulyana Trafimovich7a20cbd2019-11-08 11:27:31 +0000324 dir = '%s/%s' % (path, arch)
325 found, _ = self.is_dir(dir)
326 if found:
327 dirs.append(dir)
328 return dirs
329
330 def check_art_test_executable(self, filename):
331 dirs = self.arch_dirs_for_path(ART_TEST_DIR)
332 if not dirs:
Roland Levillain61f07162019-06-26 12:44:04 +0100333 self.fail('ART test binary missing: %s', filename)
Ulyana Trafimovich7a20cbd2019-11-08 11:27:31 +0000334 for dir in dirs:
335 test_path = '%s/%s' % (dir, filename)
336 self._expected_file_globs.add(test_path)
337 if not self._provider.get(test_path).is_exec:
338 self.fail('%s is not executable', test_path)
Roland Levillain61f07162019-06-26 12:44:04 +0100339
David Srbecky4a88a5a2020-05-05 16:21:57 +0100340 def check_art_test_data(self, filename):
341 dirs = self.arch_dirs_for_path(ART_TEST_DIR)
342 if not dirs:
343 self.fail('ART test data missing: %s', filename)
344 for dir in dirs:
345 if not self.check_file('%s/%s' % (dir, filename)):
346 return
347
Martin Stjernholm2babede2019-03-18 21:04:49 +0000348 def check_single_library(self, filename):
349 lib_path = 'lib/%s' % filename
350 lib64_path = 'lib64/%s' % filename
351 lib_is_file, _ = self.is_file(lib_path)
352 if lib_is_file:
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000353 self._expected_file_globs.add(lib_path)
Martin Stjernholm2babede2019-03-18 21:04:49 +0000354 lib64_is_file, _ = self.is_file(lib64_path)
355 if lib64_is_file:
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000356 self._expected_file_globs.add(lib64_path)
Martin Stjernholm2babede2019-03-18 21:04:49 +0000357 if not lib_is_file and not lib64_is_file:
358 self.fail('Library missing: %s', filename)
Andreas Gampea0242cf2019-01-29 13:01:23 -0800359
Ulyana Trafimovich7a20cbd2019-11-08 11:27:31 +0000360 def check_dexpreopt(self, basename):
361 dirs = self.arch_dirs_for_path('javalib')
Ulyana Trafimovich7a20cbd2019-11-08 11:27:31 +0000362 for dir in dirs:
Tim Joinesbe50bc32020-03-18 18:00:41 +0000363 for ext in ['art', 'oat', 'vdex']:
Ulyana Trafimovich7a20cbd2019-11-08 11:27:31 +0000364 self.check_file('%s/%s.%s' % (dir, basename, ext))
365
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000366 def check_java_library(self, basename):
367 return self.check_file('javalib/%s.jar' % basename)
Andreas Gampe09123952019-01-30 13:17:02 -0800368
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000369 def ignore_path(self, path_glob):
370 self._expected_file_globs.add(path_glob)
Martin Stjernholm2babede2019-03-18 21:04:49 +0000371
Roland Levillain61f07162019-06-26 12:44:04 +0100372 def check_optional_art_test_executable(self, filename):
Orion Hodsonaf3ed622019-10-03 16:30:54 +0100373 for arch in ARCHS:
374 self.ignore_path('%s/%s/%s' % (ART_TEST_DIR, arch, filename))
Roland Levillain61f07162019-06-26 12:44:04 +0100375
Martin Stjernholm2babede2019-03-18 21:04:49 +0000376 def check_no_superfluous_files(self, dir_path):
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000377 paths = []
Martin Stjernholm2babede2019-03-18 21:04:49 +0000378 for name in sorted(self._provider.read_dir(dir_path).keys()):
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000379 if name not in ('.', '..'):
380 paths.append(os.path.join(dir_path, name))
381 expected_paths = set()
382 dir_prefix = dir_path + '/'
383 for path_glob in self._expected_file_globs:
384 expected_paths |= set(fnmatch.filter(paths, path_glob))
385 # If there are globs in subdirectories of dir_path we want to match their
386 # path segments at this directory level.
387 if path_glob.startswith(dir_prefix):
388 subpath = path_glob[len(dir_prefix):]
389 subpath_first_segment, _, _ = subpath.partition('/')
390 expected_paths |= set(fnmatch.filter(paths, dir_prefix + subpath_first_segment))
391 for unexpected_path in set(paths) - expected_paths:
392 self.fail('Unexpected file \'%s\'', unexpected_path)
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000393
Andreas Gampeeb555b82019-01-30 14:47:49 -0800394 # Just here for docs purposes, even if it isn't good Python style.
395
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000396 def check_symlinked_multilib_executable(self, filename):
397 """Check bin/filename32, and/or bin/filename64, with symlink bin/filename."""
Andreas Gampeeb555b82019-01-30 14:47:49 -0800398 raise NotImplementedError
399
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100400 def check_symlinked_first_executable(self, filename):
401 """Check bin/filename32, and/or bin/filename64, with symlink bin/filename."""
402 raise NotImplementedError
403
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000404 def check_multilib_executable(self, filename):
405 """Check bin/filename for 32 bit, and/or bin/filename64."""
Andreas Gampeeb555b82019-01-30 14:47:49 -0800406 raise NotImplementedError
407
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100408 def check_first_executable(self, filename):
409 """Check bin/filename for 32 bit, and/or bin/filename64."""
410 raise NotImplementedError
411
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000412 def check_native_library(self, basename):
413 """Check lib/basename.so, and/or lib64/basename.so."""
Martin Stjernholm2babede2019-03-18 21:04:49 +0000414 raise NotImplementedError
415
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000416 def check_optional_native_library(self, basename_glob):
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000417 """Allow lib/basename.so and/or lib64/basename.so to exist."""
Martin Stjernholmb81fe232019-03-25 17:38:04 +0000418 raise NotImplementedError
419
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000420 def check_prefer64_library(self, basename):
421 """Check lib64/basename.so, or lib/basename.so on 32 bit only."""
Andreas Gampeeb555b82019-01-30 14:47:49 -0800422 raise NotImplementedError
423
424
425class Arch32Checker(Checker):
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000426 def check_symlinked_multilib_executable(self, filename):
427 self.check_executable('%s32' % filename)
428 self.check_executable_symlink(filename)
Andreas Gampeeb555b82019-01-30 14:47:49 -0800429
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100430 def check_symlinked_first_executable(self, filename):
431 self.check_executable('%s32' % filename)
432 self.check_executable_symlink(filename)
433
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000434 def check_multilib_executable(self, filename):
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100435 self.check_executable('%s32' % filename)
436
437 def check_first_executable(self, filename):
438 self.check_executable('%s32' % filename)
Martin Stjernholm2babede2019-03-18 21:04:49 +0000439
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000440 def check_native_library(self, basename):
Andreas Gampeeb555b82019-01-30 14:47:49 -0800441 # TODO: Use $TARGET_ARCH (e.g. check whether it is "arm" or "arm64") to improve
442 # the precision of this test?
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000443 self.check_file('lib/%s.so' % basename)
Andreas Gampeeb555b82019-01-30 14:47:49 -0800444
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000445 def check_optional_native_library(self, basename_glob):
446 self.ignore_path('lib/%s.so' % basename_glob)
Martin Stjernholmb81fe232019-03-25 17:38:04 +0000447
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000448 def check_prefer64_library(self, basename):
449 self.check_native_library(basename)
Andreas Gampeeb555b82019-01-30 14:47:49 -0800450
451
452class Arch64Checker(Checker):
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000453 def check_symlinked_multilib_executable(self, filename):
454 self.check_executable('%s64' % filename)
455 self.check_executable_symlink(filename)
Andreas Gampeeb555b82019-01-30 14:47:49 -0800456
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100457 def check_symlinked_first_executable(self, filename):
458 self.check_executable('%s64' % filename)
459 self.check_executable_symlink(filename)
460
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000461 def check_multilib_executable(self, filename):
462 self.check_executable('%s64' % filename)
Martin Stjernholm2babede2019-03-18 21:04:49 +0000463
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100464 def check_first_executable(self, filename):
465 self.check_executable('%s64' % filename)
466
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000467 def check_native_library(self, basename):
Andreas Gampeeb555b82019-01-30 14:47:49 -0800468 # TODO: Use $TARGET_ARCH (e.g. check whether it is "arm" or "arm64") to improve
469 # the precision of this test?
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000470 self.check_file('lib64/%s.so' % basename)
Andreas Gampeeb555b82019-01-30 14:47:49 -0800471
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000472 def check_optional_native_library(self, basename_glob):
473 self.ignore_path('lib64/%s.so' % basename_glob)
Martin Stjernholmb81fe232019-03-25 17:38:04 +0000474
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000475 def check_prefer64_library(self, basename):
476 self.check_native_library(basename)
Andreas Gampeeb555b82019-01-30 14:47:49 -0800477
478
479class MultilibChecker(Checker):
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000480 def check_symlinked_multilib_executable(self, filename):
481 self.check_executable('%s32' % filename)
482 self.check_executable('%s64' % filename)
483 self.check_executable_symlink(filename)
Andreas Gampeeb555b82019-01-30 14:47:49 -0800484
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100485 def check_symlinked_first_executable(self, filename):
486 self.check_executable('%s64' % filename)
487 self.check_executable_symlink(filename)
488
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000489 def check_multilib_executable(self, filename):
490 self.check_executable('%s64' % filename)
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100491 self.check_executable('%s32' % filename)
492
493 def check_first_executable(self, filename):
494 self.check_executable('%s64' % filename)
Martin Stjernholm2babede2019-03-18 21:04:49 +0000495
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000496 def check_native_library(self, basename):
Andreas Gampeeb555b82019-01-30 14:47:49 -0800497 # TODO: Use $TARGET_ARCH (e.g. check whether it is "arm" or "arm64") to improve
498 # the precision of this test?
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000499 self.check_file('lib/%s.so' % basename)
500 self.check_file('lib64/%s.so' % basename)
Andreas Gampeeb555b82019-01-30 14:47:49 -0800501
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000502 def check_optional_native_library(self, basename_glob):
503 self.ignore_path('lib/%s.so' % basename_glob)
504 self.ignore_path('lib64/%s.so' % basename_glob)
Martin Stjernholmb81fe232019-03-25 17:38:04 +0000505
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000506 def check_prefer64_library(self, basename):
507 self.check_file('lib64/%s.so' % basename)
Andreas Gampeeb555b82019-01-30 14:47:49 -0800508
509
Andreas Gampe9dc4b052019-01-30 13:47:25 -0800510class ReleaseChecker:
511 def __init__(self, checker):
512 self._checker = checker
Luca Stefani4e91ee92019-03-06 15:08:16 +0100513
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000514 def __str__(self):
515 return 'Release Checker'
516
517 def run(self):
Roland Levillaindf66de02019-11-11 23:22:15 +0000518 # Check the Protocol Buffers APEX manifest.
519 self._checker.check_file('apex_manifest.pb')
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000520
Martin Stjernholm77f17662019-04-03 17:08:21 +0100521 # Check binaries for ART.
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100522 self._checker.check_first_executable('dex2oat')
Martin Stjernholm77f17662019-04-03 17:08:21 +0100523 self._checker.check_executable('dexdump')
524 self._checker.check_executable('dexlist')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000525 self._checker.check_executable('dexoptanalyzer')
526 self._checker.check_executable('profman')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000527 self._checker.check_symlinked_multilib_executable('dalvikvm')
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000528
Martin Stjernholm023c2182019-03-28 15:52:32 +0000529 # Check exported libraries for ART.
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000530 self._checker.check_native_library('libdexfile_external')
531 self._checker.check_native_library('libnativebridge')
532 self._checker.check_native_library('libnativehelper')
533 self._checker.check_native_library('libnativeloader')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000534
535 # Check internal libraries for ART.
536 self._checker.check_native_library('libadbconnection')
537 self._checker.check_native_library('libart')
538 self._checker.check_native_library('libart-compiler')
539 self._checker.check_native_library('libart-dexlayout')
Roland Levillain5b768892020-02-19 15:49:02 +0000540 self._checker.check_native_library('libart-disassembler')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000541 self._checker.check_native_library('libartbase')
542 self._checker.check_native_library('libartpalette')
543 self._checker.check_native_library('libdexfile')
544 self._checker.check_native_library('libdexfile_support')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000545 self._checker.check_native_library('libopenjdkjvm')
546 self._checker.check_native_library('libopenjdkjvmti')
547 self._checker.check_native_library('libprofile')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000548 self._checker.check_native_library('libsigchain')
Ulyana Trafimoviche886d682020-07-16 15:09:38 +0000549 # Only on ARM/ARM64
550 self._checker.check_optional_native_library('libart-simulator-container')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000551
552 # Check java libraries for Managed Core Library.
553 self._checker.check_java_library('apache-xml')
554 self._checker.check_java_library('bouncycastle')
555 self._checker.check_java_library('core-libart')
556 self._checker.check_java_library('core-oj')
557 self._checker.check_java_library('okhttp')
Ulya Trafimovich7b0df592020-01-03 13:29:00 +0000558 if isEnvTrue('EMMA_INSTRUMENT_FRAMEWORK'):
559 # In coverage builds jacoco is added to the list of ART apex jars.
560 self._checker.check_java_library('jacocoagent')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000561
562 # Check internal native libraries for Managed Core Library.
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000563 self._checker.check_native_library('libjavacore')
564 self._checker.check_native_library('libopenjdk')
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000565
Martin Stjernholm2babede2019-03-18 21:04:49 +0000566 # Check internal native library dependencies.
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000567 #
Martin Stjernholm2babede2019-03-18 21:04:49 +0000568 # Any internal dependency not listed here will cause a failure in
569 # NoSuperfluousLibrariesChecker. Internal dependencies are generally just
570 # implementation details, but in the release package we want to track them
571 # because a) they add to the package size and the RAM usage (in particular
572 # if the library is also present in /system or another APEX and hence might
573 # get loaded twice through linker namespace separation), and b) we need to
574 # catch invalid dependencies on /system or other APEXes that should go
575 # through an exported library with stubs (b/128708192 tracks implementing a
576 # better approach for that).
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000577 self._checker.check_native_library('libbacktrace')
578 self._checker.check_native_library('libbase')
579 self._checker.check_native_library('libc++')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000580 self._checker.check_native_library('libdt_fd_forward')
581 self._checker.check_native_library('libdt_socket')
582 self._checker.check_native_library('libjdwp')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000583 self._checker.check_native_library('liblzma')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000584 self._checker.check_native_library('libnpt')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000585 self._checker.check_native_library('libunwindstack')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000586 self._checker.check_native_library('libziparchive')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000587 self._checker.check_optional_native_library('libvixl') # Only on ARM/ARM64
Martin Stjernholm2babede2019-03-18 21:04:49 +0000588
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000589 # Allow extra dependencies that appear in ASAN builds.
590 self._checker.check_optional_native_library('libclang_rt.asan*')
591 self._checker.check_optional_native_library('libclang_rt.hwasan*')
592 self._checker.check_optional_native_library('libclang_rt.ubsan*')
593
Ulya Trafimovich7b0df592020-01-03 13:29:00 +0000594 # Check dexpreopt files for libcore bootclasspath jars.
595 self._checker.check_dexpreopt('boot')
596 self._checker.check_dexpreopt('boot-apache-xml')
597 self._checker.check_dexpreopt('boot-bouncycastle')
Ulya Trafimovich7b0df592020-01-03 13:29:00 +0000598 self._checker.check_dexpreopt('boot-core-libart')
599 self._checker.check_dexpreopt('boot-okhttp')
600 if isEnvTrue('EMMA_INSTRUMENT_FRAMEWORK'):
601 # In coverage builds the ART boot image includes jacoco.
602 self._checker.check_dexpreopt('boot-jacocoagent')
Luca Stefani4e91ee92019-03-06 15:08:16 +0100603
Andreas Gampe9dc4b052019-01-30 13:47:25 -0800604class ReleaseTargetChecker:
605 def __init__(self, checker):
606 self._checker = checker
Luca Stefani4e91ee92019-03-06 15:08:16 +0100607
Andreas Gampe09123952019-01-30 13:17:02 -0800608 def __str__(self):
609 return 'Release (Target) Checker'
610
611 def run(self):
Roland Levillaindf66de02019-11-11 23:22:15 +0000612 # We don't check for the presence of the JSON APEX manifest (file
613 # `apex_manifest.json`, only present in target APEXes), as it is only
614 # included for compatibility reasons with Android Q and will likely be
615 # removed in Android R.
616
Martin Stjernholm023c2182019-03-28 15:52:32 +0000617 # Check binaries for ART.
Martin Stjernholm023c2182019-03-28 15:52:32 +0000618 self._checker.check_executable('oatdump')
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100619 self._checker.check_multilib_executable('dex2oat')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000620
621 # Check internal libraries for ART.
622 self._checker.check_prefer64_library('libart-disassembler')
Florian Mayer07710c52019-09-16 15:53:38 +0000623 self._checker.check_native_library('libperfetto_hprof')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000624
Martin Stjernholm023c2182019-03-28 15:52:32 +0000625 # Check exported native libraries for Managed Core Library.
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000626 self._checker.check_native_library('libandroidio')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000627
628 # Check internal native library dependencies.
629 self._checker.check_native_library('libcrypto')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000630 self._checker.check_native_library('libexpat')
Martin Stjernholm2babede2019-03-18 21:04:49 +0000631
Martin Stjernholmec2db5f2019-09-04 20:54:57 +0100632 # TODO(b/139046641): Fix proper 2nd arch checks. For now, just ignore these
633 # directories.
634 self._checker.ignore_path('bin/arm')
635 self._checker.ignore_path('lib/arm')
636 self._checker.ignore_path('lib64/arm')
637
Luca Stefani4e91ee92019-03-06 15:08:16 +0100638
Andreas Gampe9dc4b052019-01-30 13:47:25 -0800639class ReleaseHostChecker:
640 def __init__(self, checker):
Luca Stefani4e91ee92019-03-06 15:08:16 +0100641 self._checker = checker
642
Andreas Gampe09123952019-01-30 13:17:02 -0800643 def __str__(self):
644 return 'Release (Host) Checker'
645
646 def run(self):
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000647 # Check binaries for ART.
648 self._checker.check_executable('hprof-conv')
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100649 self._checker.check_symlinked_first_executable('dex2oatd')
650 self._checker.check_symlinked_first_executable('dex2oat')
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000651
Martin Stjernholm023c2182019-03-28 15:52:32 +0000652 # Check exported native libraries for Managed Core Library.
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000653 self._checker.check_native_library('libandroidicu-host')
654 self._checker.check_native_library('libandroidio')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000655
656 # Check internal libraries for Managed Core Library.
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000657 self._checker.check_native_library('libexpat-host')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000658 self._checker.check_native_library('libz-host')
Andreas Gampe09123952019-01-30 13:17:02 -0800659
Luca Stefani4e91ee92019-03-06 15:08:16 +0100660
Andreas Gampe9dc4b052019-01-30 13:47:25 -0800661class DebugChecker:
662 def __init__(self, checker):
663 self._checker = checker
Luca Stefani4e91ee92019-03-06 15:08:16 +0100664
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000665 def __str__(self):
666 return 'Debug Checker'
667
668 def run(self):
Martin Stjernholm023c2182019-03-28 15:52:32 +0000669 # Check binaries for ART.
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000670 self._checker.check_executable('dexdiag')
Roland Levillain6dd62122019-06-18 19:10:21 +0100671 self._checker.check_executable('dexanalyze')
672 self._checker.check_executable('dexlayout')
673 self._checker.check_symlinked_multilib_executable('imgdiag')
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000674
Martin Stjernholm023c2182019-03-28 15:52:32 +0000675 # Check debug binaries for ART.
Roland Levillain6dd62122019-06-18 19:10:21 +0100676 self._checker.check_executable('dexlayoutd')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000677 self._checker.check_executable('dexoptanalyzerd')
Roland Levillain6dd62122019-06-18 19:10:21 +0100678 self._checker.check_symlinked_multilib_executable('imgdiagd')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000679 self._checker.check_executable('profmand')
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000680
Martin Stjernholm023c2182019-03-28 15:52:32 +0000681 # Check internal libraries for ART.
682 self._checker.check_native_library('libadbconnectiond')
Roland Levillain4d6f24d2019-10-02 16:47:36 +0100683 self._checker.check_native_library('libart-disassembler')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000684 self._checker.check_native_library('libartbased')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000685 self._checker.check_native_library('libartd')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000686 self._checker.check_native_library('libartd-compiler')
687 self._checker.check_native_library('libartd-dexlayout')
Roland Levillain4d6f24d2019-10-02 16:47:36 +0100688 self._checker.check_native_library('libartd-disassembler')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000689 self._checker.check_native_library('libdexfiled')
690 self._checker.check_native_library('libopenjdkjvmd')
691 self._checker.check_native_library('libopenjdkjvmtid')
692 self._checker.check_native_library('libprofiled')
Ulyana Trafimoviche886d682020-07-16 15:09:38 +0000693 # Only on ARM/ARM64
694 self._checker.check_optional_native_library('libartd-simulator-container')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000695
696 # Check internal libraries for Managed Core Library.
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000697 self._checker.check_native_library('libopenjdkd')
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000698
Luca Stefani4e91ee92019-03-06 15:08:16 +0100699
Andreas Gampe9dc4b052019-01-30 13:47:25 -0800700class DebugTargetChecker:
701 def __init__(self, checker):
702 self._checker = checker
Luca Stefani4e91ee92019-03-06 15:08:16 +0100703
Andreas Gampea0242cf2019-01-29 13:01:23 -0800704 def __str__(self):
705 return 'Debug (Target) Checker'
706
707 def run(self):
Martin Stjernholm023c2182019-03-28 15:52:32 +0000708 # Check ART debug binaries.
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100709 self._checker.check_multilib_executable('dex2oatd')
710 self._checker.check_multilib_executable('dex2oat')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000711 self._checker.check_executable('oatdumpd')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000712
713 # Check ART internal libraries.
Martin Stjernholm356864a2019-04-30 16:22:17 +0100714 self._checker.check_native_library('libdexfiled_external')
Florian Mayer07710c52019-09-16 15:53:38 +0000715 self._checker.check_native_library('libperfetto_hprofd')
Andreas Gampea0242cf2019-01-29 13:01:23 -0800716
Martin Stjernholm2babede2019-03-18 21:04:49 +0000717 # Check internal native library dependencies.
718 #
719 # Like in the release package, we check that we don't get other dependencies
720 # besides those listed here. In this case the concern is not bloat, but
721 # rather that we don't get behavioural differences between user (release)
722 # and userdebug/eng builds, which could happen if the debug package has
723 # duplicate library instances where releases don't. In other words, it's
724 # uncontroversial to add debug-only dependencies, as long as they don't make
725 # assumptions on having a single global state (ideally they should have
726 # double_loadable:true, cf. go/double_loadable). Also, like in the release
727 # package we need to look out for dependencies that should go through
728 # exported library stubs (until b/128708192 is fixed).
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000729 self._checker.check_optional_native_library('libvixld') # Only on ARM/ARM64
Martin Stjernholm77f17662019-04-03 17:08:21 +0100730 self._checker.check_prefer64_library('libmeminfo')
731 self._checker.check_prefer64_library('libprocinfo')
Martin Stjernholm2babede2019-03-18 21:04:49 +0000732
733
Roland Levillain61f07162019-06-26 12:44:04 +0100734class TestingTargetChecker:
735 def __init__(self, checker):
736 self._checker = checker
737
738 def __str__(self):
739 return 'Testing (Target) Checker'
740
741 def run(self):
David Srbecky883c1342020-05-11 23:30:29 +0000742 # Check ART test binaries.
743 self._checker.check_art_test_executable('art_cmdline_tests')
744 self._checker.check_art_test_executable('art_compiler_tests')
745 self._checker.check_art_test_executable('art_dex2oat_tests')
746 self._checker.check_art_test_executable('art_dexanalyze_tests')
747 self._checker.check_art_test_executable('art_dexdiag_tests')
748 self._checker.check_art_test_executable('art_dexdump_tests')
749 self._checker.check_art_test_executable('art_dexlayout_tests')
750 self._checker.check_art_test_executable('art_dexlist_tests')
751 self._checker.check_art_test_executable('art_dexoptanalyzer_tests')
752 self._checker.check_art_test_executable('art_imgdiag_tests')
753 self._checker.check_art_test_executable('art_libartbase_tests')
754 self._checker.check_art_test_executable('art_libartpalette_tests')
Roland Levillain61f07162019-06-26 12:44:04 +0100755 self._checker.check_art_test_executable('art_libdexfile_support_tests')
David Srbecky883c1342020-05-11 23:30:29 +0000756 self._checker.check_art_test_executable('art_libdexfile_tests')
757 self._checker.check_art_test_executable('art_libprofile_tests')
758 self._checker.check_art_test_executable('art_oatdump_tests')
759 self._checker.check_art_test_executable('art_profman_tests')
760 self._checker.check_art_test_executable('art_runtime_compiler_tests')
761 self._checker.check_art_test_executable('art_runtime_tests')
762 self._checker.check_art_test_executable('art_sigchain_tests')
Roland Levillain61f07162019-06-26 12:44:04 +0100763
764 # Check ART test (internal) libraries.
765 self._checker.check_native_library('libart-gtest')
Roland Levillain61f07162019-06-26 12:44:04 +0100766
Andreas Gampe816a5522019-09-11 11:29:49 -0700767 # Check ART test tools.
768 self._checker.check_executable('signal_dumper')
769
David Srbecky4a88a5a2020-05-05 16:21:57 +0100770 # Check ART jar files which are needed for gtests.
771 self._checker.check_art_test_data('art-gtest-jars-AbstractMethod.jar')
772 self._checker.check_art_test_data('art-gtest-jars-MyClassNatives.jar')
773 self._checker.check_art_test_data('art-gtest-jars-Main.jar')
774 self._checker.check_art_test_data('art-gtest-jars-ProtoCompare.jar')
775 self._checker.check_art_test_data('art-gtest-jars-Transaction.jar')
David Srbeckyedda3ca2020-07-08 00:41:08 +0100776 self._checker.check_art_test_data('art-gtest-jars-VerifierDepsMulti.dex')
David Srbecky4a88a5a2020-05-05 16:21:57 +0100777 self._checker.check_art_test_data('art-gtest-jars-Nested.jar')
778 self._checker.check_art_test_data('art-gtest-jars-MyClass.jar')
779 self._checker.check_art_test_data('art-gtest-jars-ManyMethods.jar')
780 self._checker.check_art_test_data('art-gtest-jars-GetMethodSignature.jar')
781 self._checker.check_art_test_data('art-gtest-jars-Lookup.jar')
782 self._checker.check_art_test_data('art-gtest-jars-Instrumentation.jar')
783 self._checker.check_art_test_data('art-gtest-jars-MainUncompressedAligned.jar')
784 self._checker.check_art_test_data('art-gtest-jars-ForClassLoaderD.jar')
785 self._checker.check_art_test_data('art-gtest-jars-ForClassLoaderC.jar')
786 self._checker.check_art_test_data('art-gtest-jars-ErroneousA.jar')
787 self._checker.check_art_test_data('art-gtest-jars-DexToDexDecompiler.jar')
788 self._checker.check_art_test_data('art-gtest-jars-HiddenApiSignatures.jar')
789 self._checker.check_art_test_data('art-gtest-jars-ForClassLoaderB.jar')
David Srbeckyedda3ca2020-07-08 00:41:08 +0100790 self._checker.check_art_test_data('art-gtest-jars-LinkageTest.dex')
David Srbecky4a88a5a2020-05-05 16:21:57 +0100791 self._checker.check_art_test_data('art-gtest-jars-MethodTypes.jar')
792 self._checker.check_art_test_data('art-gtest-jars-ErroneousInit.jar')
David Srbeckyedda3ca2020-07-08 00:41:08 +0100793 self._checker.check_art_test_data('art-gtest-jars-VerifierDeps.dex')
David Srbecky4a88a5a2020-05-05 16:21:57 +0100794 self._checker.check_art_test_data('art-gtest-jars-StringLiterals.jar')
795 self._checker.check_art_test_data('art-gtest-jars-XandY.jar')
796 self._checker.check_art_test_data('art-gtest-jars-ExceptionHandle.jar')
797 self._checker.check_art_test_data('art-gtest-jars-ImageLayoutB.jar')
798 self._checker.check_art_test_data('art-gtest-jars-Interfaces.jar')
799 self._checker.check_art_test_data('art-gtest-jars-IMTB.jar')
800 self._checker.check_art_test_data('art-gtest-jars-Extension2.jar')
801 self._checker.check_art_test_data('art-gtest-jars-Extension1.jar')
802 self._checker.check_art_test_data('art-gtest-jars-MainEmptyUncompressedAligned.jar')
803 self._checker.check_art_test_data('art-gtest-jars-ErroneousB.jar')
804 self._checker.check_art_test_data('art-gtest-jars-MultiDexModifiedSecondary.jar')
805 self._checker.check_art_test_data('art-gtest-jars-NonStaticLeafMethods.jar')
806 self._checker.check_art_test_data('art-gtest-jars-DefaultMethods.jar')
807 self._checker.check_art_test_data('art-gtest-jars-MultiDexUncompressedAligned.jar')
808 self._checker.check_art_test_data('art-gtest-jars-StaticsFromCode.jar')
809 self._checker.check_art_test_data('art-gtest-jars-ProfileTestMultiDex.jar')
David Srbeckyedda3ca2020-07-08 00:41:08 +0100810 self._checker.check_art_test_data('art-gtest-jars-VerifySoftFailDuringClinit.dex')
David Srbecky4a88a5a2020-05-05 16:21:57 +0100811 self._checker.check_art_test_data('art-gtest-jars-MainStripped.jar')
812 self._checker.check_art_test_data('art-gtest-jars-ForClassLoaderA.jar')
813 self._checker.check_art_test_data('art-gtest-jars-StaticLeafMethods.jar')
814 self._checker.check_art_test_data('art-gtest-jars-MultiDex.jar')
815 self._checker.check_art_test_data('art-gtest-jars-Packages.jar')
816 self._checker.check_art_test_data('art-gtest-jars-ProtoCompare2.jar')
817 self._checker.check_art_test_data('art-gtest-jars-Statics.jar')
818 self._checker.check_art_test_data('art-gtest-jars-AllFields.jar')
819 self._checker.check_art_test_data('art-gtest-jars-IMTA.jar')
820 self._checker.check_art_test_data('art-gtest-jars-ImageLayoutA.jar')
821 self._checker.check_art_test_data('art-gtest-jars-MainEmptyUncompressed.jar')
822
Roland Levillain61f07162019-06-26 12:44:04 +0100823
Martin Stjernholm2babede2019-03-18 21:04:49 +0000824class NoSuperfluousBinariesChecker:
825 def __init__(self, checker):
826 self._checker = checker
827
828 def __str__(self):
829 return 'No superfluous binaries checker'
830
831 def run(self):
832 self._checker.check_no_superfluous_files('bin')
833
834
835class NoSuperfluousLibrariesChecker:
836 def __init__(self, checker):
837 self._checker = checker
838
839 def __str__(self):
840 return 'No superfluous libraries checker'
841
842 def run(self):
843 self._checker.check_no_superfluous_files('javalib')
844 self._checker.check_no_superfluous_files('lib')
Martin Stjernholm2babede2019-03-18 21:04:49 +0000845 self._checker.check_no_superfluous_files('lib64')
Martin Stjernholm2babede2019-03-18 21:04:49 +0000846
Andreas Gampeb1d55672019-01-29 22:17:02 +0000847
Roland Levillain61f07162019-06-26 12:44:04 +0100848class NoSuperfluousArtTestsChecker:
849 def __init__(self, checker):
850 self._checker = checker
851
852 def __str__(self):
853 return 'No superfluous ART tests checker'
854
855 def run(self):
Orion Hodsonaf3ed622019-10-03 16:30:54 +0100856 for arch in ARCHS:
857 self._checker.check_no_superfluous_files('%s/%s' % (ART_TEST_DIR, arch))
Roland Levillain61f07162019-06-26 12:44:04 +0100858
859
Luca Stefani4e91ee92019-03-06 15:08:16 +0100860class List:
Roland Levillain77251f92019-08-09 15:00:04 +0100861 def __init__(self, provider, print_size=False):
Luca Stefani4e91ee92019-03-06 15:08:16 +0100862 self._provider = provider
Roland Levillaine91d7872019-07-25 18:56:06 +0100863 self._print_size = print_size
Luca Stefani4e91ee92019-03-06 15:08:16 +0100864
865 def print_list(self):
Roland Levillain09744ac2019-06-06 18:23:02 +0100866
867 def print_list_rec(path):
868 apex_map = self._provider.read_dir(path)
869 if apex_map is None:
870 return
871 apex_map = dict(apex_map)
872 if '.' in apex_map:
873 del apex_map['.']
874 if '..' in apex_map:
875 del apex_map['..']
876 for (_, val) in sorted(apex_map.items()):
877 val_path = os.path.join(path, val.name)
Roland Levillaine91d7872019-07-25 18:56:06 +0100878 if self._print_size:
879 if val.size < 0:
880 print('[ n/a ] %s' % val_path)
881 else:
882 print('[%11d] %s' % (val.size, val_path))
883 else:
884 print(val_path)
Roland Levillain09744ac2019-06-06 18:23:02 +0100885 if val.is_dir:
886 print_list_rec(val_path)
887
888 print_list_rec('')
Luca Stefani4e91ee92019-03-06 15:08:16 +0100889
890
891class Tree:
Roland Levillain77251f92019-08-09 15:00:04 +0100892 def __init__(self, provider, title, print_size=False):
Luca Stefani4e91ee92019-03-06 15:08:16 +0100893 print('%s' % title)
894 self._provider = provider
Luca Stefani4e91ee92019-03-06 15:08:16 +0100895 self._has_next_list = []
Roland Levillain88e55692019-07-25 15:57:06 +0100896 self._print_size = print_size
Luca Stefani4e91ee92019-03-06 15:08:16 +0100897
898 @staticmethod
899 def get_vertical(has_next_list):
900 string = ''
901 for v in has_next_list:
902 string += '%s ' % ('│' if v else ' ')
903 return string
904
905 @staticmethod
906 def get_last_vertical(last):
907 return '└── ' if last else '├── '
908
909 def print_tree(self):
Roland Levillain09744ac2019-06-06 18:23:02 +0100910
911 def print_tree_rec(path):
912 apex_map = self._provider.read_dir(path)
913 if apex_map is None:
914 return
915 apex_map = dict(apex_map)
916 if '.' in apex_map:
917 del apex_map['.']
918 if '..' in apex_map:
919 del apex_map['..']
920 key_list = list(sorted(apex_map.keys()))
921 for i, key in enumerate(key_list):
922 prev = self.get_vertical(self._has_next_list)
923 last = self.get_last_vertical(i == len(key_list) - 1)
924 val = apex_map[key]
Roland Levillain88e55692019-07-25 15:57:06 +0100925 if self._print_size:
926 if val.size < 0:
927 print('%s%s[ n/a ] %s' % (prev, last, val.name))
928 else:
929 print('%s%s[%11d] %s' % (prev, last, val.size, val.name))
930 else:
931 print('%s%s%s' % (prev, last, val.name))
Roland Levillain09744ac2019-06-06 18:23:02 +0100932 if val.is_dir:
933 self._has_next_list.append(i < len(key_list) - 1)
934 val_path = os.path.join(path, val.name)
935 print_tree_rec(val_path)
936 self._has_next_list.pop()
937
938 print_tree_rec('')
Luca Stefani4e91ee92019-03-06 15:08:16 +0100939
Andreas Gampeb1d55672019-01-29 22:17:02 +0000940
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000941# Note: do not sys.exit early, for __del__ cleanup.
Luca Stefani4e91ee92019-03-06 15:08:16 +0100942def art_apex_test_main(test_args):
Roland Levillaindd20d002019-07-19 16:09:47 +0100943 if test_args.host and test_args.flattened:
944 logging.error("Both of --host and --flattened set")
945 return 1
Luca Stefani4e91ee92019-03-06 15:08:16 +0100946 if test_args.list and test_args.tree:
Andreas Gampeb1d55672019-01-29 22:17:02 +0000947 logging.error("Both of --list and --tree set")
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000948 return 1
Roland Levillaine91d7872019-07-25 18:56:06 +0100949 if test_args.size and not (test_args.list or test_args.tree):
950 logging.error("--size set but neither --list nor --tree set")
Roland Levillain88e55692019-07-25 15:57:06 +0100951 return 1
Martin Stjernholm3cb59a42019-08-07 17:18:29 +0100952 if not test_args.flattened and not test_args.tmpdir:
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000953 logging.error("Need a tmpdir.")
954 return 1
Martin Stjernholm3cb59a42019-08-07 17:18:29 +0100955 if not test_args.flattened and not test_args.host and not test_args.debugfs:
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000956 logging.error("Need debugfs.")
957 return 1
958
Roland Levillainfdbe2072019-11-11 15:52:45 +0000959 if test_args.host:
960 # Host APEX.
961 if test_args.flavor not in [FLAVOR_DEBUG, FLAVOR_AUTO]:
962 logging.error("Using option --host with non-Debug APEX")
Roland Levillaindbd6a142019-10-03 15:08:28 +0100963 return 1
Roland Levillainfdbe2072019-11-11 15:52:45 +0000964 # Host APEX is always a debug flavor (for now).
965 test_args.flavor = FLAVOR_DEBUG
966 else:
967 # Device APEX.
968 if test_args.flavor == FLAVOR_AUTO:
969 logging.warning('--flavor=auto, trying to autodetect. This may be incorrect!')
970 for flavor in [ FLAVOR_RELEASE, FLAVOR_DEBUG, FLAVOR_TESTING ]:
971 flavor_pattern = '*.%s*' % flavor
972 if fnmatch.fnmatch(test_args.apex, flavor_pattern):
973 test_args.flavor = flavor
974 break
975 if test_args.flavor == FLAVOR_AUTO:
976 logging.error(' Could not detect APEX flavor, neither \'%s\', \'%s\' nor \'%s\' in \'%s\'',
977 FLAVOR_RELEASE, FLAVOR_DEBUG, FLAVOR_TESTING, test_args.apex)
978 return 1
Roland Levillaindbd6a142019-10-03 15:08:28 +0100979
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000980 try:
Luca Stefani4e91ee92019-03-06 15:08:16 +0100981 if test_args.host:
982 apex_provider = HostApexProvider(test_args.apex, test_args.tmpdir)
Andreas Gampe09123952019-01-30 13:17:02 -0800983 else:
Roland Levillaindd20d002019-07-19 16:09:47 +0100984 if test_args.flattened:
985 apex_provider = TargetFlattenedApexProvider(test_args.apex)
986 else:
987 apex_provider = TargetApexProvider(test_args.apex, test_args.tmpdir, test_args.debugfs)
Luca Stefani4e91ee92019-03-06 15:08:16 +0100988 except (zipfile.BadZipFile, zipfile.LargeZipFile) as e:
Andreas Gampea0242cf2019-01-29 13:01:23 -0800989 logging.error('Failed to create provider: %s', e)
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000990 return 1
991
Luca Stefani4e91ee92019-03-06 15:08:16 +0100992 if test_args.tree:
Roland Levillain88e55692019-07-25 15:57:06 +0100993 Tree(apex_provider, test_args.apex, test_args.size).print_tree()
Andreas Gampeb1d55672019-01-29 22:17:02 +0000994 return 0
Luca Stefani4e91ee92019-03-06 15:08:16 +0100995 if test_args.list:
Roland Levillaine91d7872019-07-25 18:56:06 +0100996 List(apex_provider, test_args.size).print_list()
Andreas Gampeb1d55672019-01-29 22:17:02 +0000997 return 0
998
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000999 checkers = []
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001000 if test_args.bitness == BITNESS_AUTO:
Luca Stefani4e91ee92019-03-06 15:08:16 +01001001 logging.warning('--bitness=auto, trying to autodetect. This may be incorrect!')
Andreas Gampeeb555b82019-01-30 14:47:49 -08001002 has_32 = apex_provider.get('lib') is not None
1003 has_64 = apex_provider.get('lib64') is not None
1004 if has_32 and has_64:
Luca Stefani4e91ee92019-03-06 15:08:16 +01001005 logging.warning(' Detected multilib')
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001006 test_args.bitness = BITNESS_MULTILIB
Andreas Gampeeb555b82019-01-30 14:47:49 -08001007 elif has_32:
Luca Stefani4e91ee92019-03-06 15:08:16 +01001008 logging.warning(' Detected 32-only')
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001009 test_args.bitness = BITNESS_32
Andreas Gampeeb555b82019-01-30 14:47:49 -08001010 elif has_64:
Luca Stefani4e91ee92019-03-06 15:08:16 +01001011 logging.warning(' Detected 64-only')
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001012 test_args.bitness = BITNESS_64
Andreas Gampeeb555b82019-01-30 14:47:49 -08001013 else:
1014 logging.error(' Could not detect bitness, neither lib nor lib64 contained.')
Roland Levillain77251f92019-08-09 15:00:04 +01001015 List(apex_provider).print_list()
Andreas Gampeeb555b82019-01-30 14:47:49 -08001016 return 1
1017
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001018 if test_args.bitness == BITNESS_32:
Andreas Gampeeb555b82019-01-30 14:47:49 -08001019 base_checker = Arch32Checker(apex_provider)
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001020 elif test_args.bitness == BITNESS_64:
Andreas Gampeeb555b82019-01-30 14:47:49 -08001021 base_checker = Arch64Checker(apex_provider)
1022 else:
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001023 assert test_args.bitness == BITNESS_MULTILIB
Andreas Gampeeb555b82019-01-30 14:47:49 -08001024 base_checker = MultilibChecker(apex_provider)
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001025
Andreas Gampe9dc4b052019-01-30 13:47:25 -08001026 checkers.append(ReleaseChecker(base_checker))
Luca Stefani4e91ee92019-03-06 15:08:16 +01001027 if test_args.host:
Andreas Gampe9dc4b052019-01-30 13:47:25 -08001028 checkers.append(ReleaseHostChecker(base_checker))
Andreas Gampe09123952019-01-30 13:17:02 -08001029 else:
Andreas Gampe9dc4b052019-01-30 13:47:25 -08001030 checkers.append(ReleaseTargetChecker(base_checker))
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001031 if test_args.flavor == FLAVOR_DEBUG or test_args.flavor == FLAVOR_TESTING:
Andreas Gampe9dc4b052019-01-30 13:47:25 -08001032 checkers.append(DebugChecker(base_checker))
Roland Levillain61f07162019-06-26 12:44:04 +01001033 if not test_args.host:
1034 checkers.append(DebugTargetChecker(base_checker))
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001035 if test_args.flavor == FLAVOR_TESTING:
Roland Levillain61f07162019-06-26 12:44:04 +01001036 checkers.append(TestingTargetChecker(base_checker))
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001037
Martin Stjernholm2babede2019-03-18 21:04:49 +00001038 # These checkers must be last.
1039 checkers.append(NoSuperfluousBinariesChecker(base_checker))
Roland Levillain61f07162019-06-26 12:44:04 +01001040 checkers.append(NoSuperfluousArtTestsChecker(base_checker))
Martin Stjernholm2babede2019-03-18 21:04:49 +00001041 if not test_args.host:
1042 # We only care about superfluous libraries on target, where their absence
1043 # can be vital to ensure they get picked up from the right package.
1044 checkers.append(NoSuperfluousLibrariesChecker(base_checker))
1045
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001046 failed = False
1047 for checker in checkers:
1048 logging.info('%s...', checker)
1049 checker.run()
Andreas Gampe9dc4b052019-01-30 13:47:25 -08001050 if base_checker.error_count() > 0:
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001051 logging.error('%s FAILED', checker)
1052 failed = True
1053 else:
1054 logging.info('%s SUCCEEDED', checker)
Andreas Gampe9dc4b052019-01-30 13:47:25 -08001055 base_checker.reset_errors()
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001056
1057 return 1 if failed else 0
1058
Luca Stefani4e91ee92019-03-06 15:08:16 +01001059
1060def art_apex_test_default(test_parser):
1061 if 'ANDROID_PRODUCT_OUT' not in os.environ:
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001062 logging.error('No-argument use requires ANDROID_PRODUCT_OUT')
1063 sys.exit(1)
1064 product_out = os.environ['ANDROID_PRODUCT_OUT']
Luca Stefani4e91ee92019-03-06 15:08:16 +01001065 if 'ANDROID_HOST_OUT' not in os.environ:
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001066 logging.error('No-argument use requires ANDROID_HOST_OUT')
1067 sys.exit(1)
1068 host_out = os.environ['ANDROID_HOST_OUT']
1069
Luca Stefani4e91ee92019-03-06 15:08:16 +01001070 test_args = test_parser.parse_args(['dummy']) # For consistency.
1071 test_args.debugfs = '%s/bin/debugfs' % host_out
1072 test_args.tmpdir = '.'
1073 test_args.tree = False
1074 test_args.list = False
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001075 test_args.bitness = BITNESS_AUTO
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001076 failed = False
1077
Luca Stefani4e91ee92019-03-06 15:08:16 +01001078 if not os.path.exists(test_args.debugfs):
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001079 logging.error("Cannot find debugfs (default path %s). Please build it, e.g., m debugfs",
Luca Stefani4e91ee92019-03-06 15:08:16 +01001080 test_args.debugfs)
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001081 sys.exit(1)
1082
Roland Levillaind7ea0432019-09-09 16:29:12 +01001083 # TODO: Add host support.
1084 # TODO: Add support for flattened APEX packages.
Luca Stefani4e91ee92019-03-06 15:08:16 +01001085 configs = [
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001086 {'name': 'com.android.art.release', 'flavor': FLAVOR_RELEASE, 'host': False},
1087 {'name': 'com.android.art.debug', 'flavor': FLAVOR_DEBUG, 'host': False},
1088 {'name': 'com.android.art.testing', 'flavor': FLAVOR_TESTING, 'host': False},
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001089 ]
1090
1091 for config in configs:
1092 logging.info(config['name'])
1093 # TODO: Host will need different path.
Luca Stefani4e91ee92019-03-06 15:08:16 +01001094 test_args.apex = '%s/system/apex/%s.apex' % (product_out, config['name'])
1095 if not os.path.exists(test_args.apex):
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001096 failed = True
Luca Stefani4e91ee92019-03-06 15:08:16 +01001097 logging.error("Cannot find APEX %s. Please build it first.", test_args.apex)
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001098 continue
Roland Levillaind7ea0432019-09-09 16:29:12 +01001099 test_args.flavor = config['flavor']
Luca Stefani4e91ee92019-03-06 15:08:16 +01001100 test_args.host = config['host']
1101 failed = art_apex_test_main(test_args) != 0
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001102
1103 if failed:
1104 sys.exit(1)
1105
Luca Stefani4e91ee92019-03-06 15:08:16 +01001106
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001107if __name__ == "__main__":
Martin Stjernholme58624f2019-09-20 15:53:40 +01001108 parser = argparse.ArgumentParser(description='Check integrity of an ART APEX.')
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001109
Roland Levillaind7ea0432019-09-09 16:29:12 +01001110 parser.add_argument('apex', help='APEX file input')
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001111
Roland Levillaind7ea0432019-09-09 16:29:12 +01001112 parser.add_argument('--host', help='Check as host APEX', action='store_true')
Andreas Gampe09123952019-01-30 13:17:02 -08001113
Roland Levillaind7ea0432019-09-09 16:29:12 +01001114 parser.add_argument('--flattened', help='Check as flattened (target) APEX', action='store_true')
Roland Levillaindd20d002019-07-19 16:09:47 +01001115
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001116 parser.add_argument('--flavor', help='Check as FLAVOR APEX', choices=FLAVORS_ALL,
1117 default=FLAVOR_AUTO)
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001118
Andreas Gampeb1d55672019-01-29 22:17:02 +00001119 parser.add_argument('--list', help='List all files', action='store_true')
1120 parser.add_argument('--tree', help='Print directory tree', action='store_true')
Roland Levillaine91d7872019-07-25 18:56:06 +01001121 parser.add_argument('--size', help='Print file sizes', action='store_true')
Andreas Gampeb1d55672019-01-29 22:17:02 +00001122
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001123 parser.add_argument('--tmpdir', help='Directory for temp files')
1124 parser.add_argument('--debugfs', help='Path to debugfs')
1125
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001126 parser.add_argument('--bitness', help='Bitness to check', choices=BITNESS_ALL,
1127 default=BITNESS_AUTO)
Andreas Gampeeb555b82019-01-30 14:47:49 -08001128
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001129 if len(sys.argv) == 1:
Luca Stefani4e91ee92019-03-06 15:08:16 +01001130 art_apex_test_default(parser)
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001131 else:
1132 args = parser.parse_args()
1133
1134 if args is None:
1135 sys.exit(1)
1136
Luca Stefani4e91ee92019-03-06 15:08:16 +01001137 exit_code = art_apex_test_main(args)
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001138 sys.exit(exit_code)