blob: 91af3db510e8b20e3d27355b4b25a4076da13223 [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')
Chris Wailesbefdad42021-01-12 16:37:57 -0800543 self._checker.check_native_library('libartservice')
544 self._checker.check_native_library('libarttools')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000545 self._checker.check_native_library('libdexfile')
546 self._checker.check_native_library('libdexfile_support')
Roland Levillain3934fe82020-08-20 20:16:42 +0100547 self._checker.check_native_library('libdt_fd_forward')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000548 self._checker.check_native_library('libopenjdkjvm')
549 self._checker.check_native_library('libopenjdkjvmti')
550 self._checker.check_native_library('libprofile')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000551 self._checker.check_native_library('libsigchain')
552
Chris Wailesbefdad42021-01-12 16:37:57 -0800553 # Check internal Java libraries
554 self._checker.check_java_library("artservice")
555
Martin Stjernholm023c2182019-03-28 15:52:32 +0000556 # Check java libraries for Managed Core Library.
557 self._checker.check_java_library('apache-xml')
558 self._checker.check_java_library('bouncycastle')
559 self._checker.check_java_library('core-libart')
560 self._checker.check_java_library('core-oj')
561 self._checker.check_java_library('okhttp')
Ulya Trafimovich7b0df592020-01-03 13:29:00 +0000562 if isEnvTrue('EMMA_INSTRUMENT_FRAMEWORK'):
563 # In coverage builds jacoco is added to the list of ART apex jars.
564 self._checker.check_java_library('jacocoagent')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000565
566 # Check internal native libraries for Managed Core Library.
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000567 self._checker.check_native_library('libjavacore')
568 self._checker.check_native_library('libopenjdk')
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000569
Martin Stjernholm2babede2019-03-18 21:04:49 +0000570 # Check internal native library dependencies.
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000571 #
Martin Stjernholm2babede2019-03-18 21:04:49 +0000572 # Any internal dependency not listed here will cause a failure in
573 # NoSuperfluousLibrariesChecker. Internal dependencies are generally just
574 # implementation details, but in the release package we want to track them
575 # because a) they add to the package size and the RAM usage (in particular
576 # if the library is also present in /system or another APEX and hence might
577 # get loaded twice through linker namespace separation), and b) we need to
578 # catch invalid dependencies on /system or other APEXes that should go
579 # through an exported library with stubs (b/128708192 tracks implementing a
580 # better approach for that).
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000581 self._checker.check_native_library('libbacktrace')
582 self._checker.check_native_library('libbase')
583 self._checker.check_native_library('libc++')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000584 self._checker.check_native_library('libdt_socket')
585 self._checker.check_native_library('libjdwp')
Martin Stjernholm2eb1f832020-09-17 00:25:52 +0100586 self._checker.check_native_library('liblz4')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000587 self._checker.check_native_library('liblzma')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000588 self._checker.check_native_library('libnpt')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000589 self._checker.check_native_library('libunwindstack')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000590 self._checker.check_native_library('libziparchive')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000591 self._checker.check_optional_native_library('libvixl') # Only on ARM/ARM64
Martin Stjernholm2babede2019-03-18 21:04:49 +0000592
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000593 # Allow extra dependencies that appear in ASAN builds.
594 self._checker.check_optional_native_library('libclang_rt.asan*')
595 self._checker.check_optional_native_library('libclang_rt.hwasan*')
596 self._checker.check_optional_native_library('libclang_rt.ubsan*')
597
Ulya Trafimovich7b0df592020-01-03 13:29:00 +0000598 # Check dexpreopt files for libcore bootclasspath jars.
599 self._checker.check_dexpreopt('boot')
600 self._checker.check_dexpreopt('boot-apache-xml')
601 self._checker.check_dexpreopt('boot-bouncycastle')
Ulya Trafimovich7b0df592020-01-03 13:29:00 +0000602 self._checker.check_dexpreopt('boot-core-libart')
603 self._checker.check_dexpreopt('boot-okhttp')
604 if isEnvTrue('EMMA_INSTRUMENT_FRAMEWORK'):
605 # In coverage builds the ART boot image includes jacoco.
606 self._checker.check_dexpreopt('boot-jacocoagent')
Luca Stefani4e91ee92019-03-06 15:08:16 +0100607
Andreas Gampe9dc4b052019-01-30 13:47:25 -0800608class ReleaseTargetChecker:
609 def __init__(self, checker):
610 self._checker = checker
Luca Stefani4e91ee92019-03-06 15:08:16 +0100611
Andreas Gampe09123952019-01-30 13:17:02 -0800612 def __str__(self):
613 return 'Release (Target) Checker'
614
615 def run(self):
Roland Levillaindf66de02019-11-11 23:22:15 +0000616 # We don't check for the presence of the JSON APEX manifest (file
617 # `apex_manifest.json`, only present in target APEXes), as it is only
618 # included for compatibility reasons with Android Q and will likely be
619 # removed in Android R.
620
Martin Stjernholm023c2182019-03-28 15:52:32 +0000621 # Check binaries for ART.
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100622 self._checker.check_multilib_executable('dex2oat')
Orion Hodson4c3ade62021-02-10 14:07:10 +0000623 self._checker.check_executable('oatdump')
624 self._checker.check_executable("odrefresh")
Martin Stjernholm023c2182019-03-28 15:52:32 +0000625
626 # Check internal libraries for ART.
Florian Mayer07710c52019-09-16 15:53:38 +0000627 self._checker.check_native_library('libperfetto_hprof')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000628
Martin Stjernholm023c2182019-03-28 15:52:32 +0000629 # Check exported native libraries for Managed Core Library.
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000630 self._checker.check_native_library('libandroidio')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000631
632 # Check internal native library dependencies.
633 self._checker.check_native_library('libcrypto')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000634 self._checker.check_native_library('libexpat')
Martin Stjernholm2babede2019-03-18 21:04:49 +0000635
Luca Stefani4e91ee92019-03-06 15:08:16 +0100636
Andreas Gampe9dc4b052019-01-30 13:47:25 -0800637class ReleaseHostChecker:
638 def __init__(self, checker):
Luca Stefani4e91ee92019-03-06 15:08:16 +0100639 self._checker = checker
640
Andreas Gampe09123952019-01-30 13:17:02 -0800641 def __str__(self):
642 return 'Release (Host) Checker'
643
644 def run(self):
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000645 # Check binaries for ART.
646 self._checker.check_executable('hprof-conv')
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100647 self._checker.check_symlinked_first_executable('dex2oatd')
648 self._checker.check_symlinked_first_executable('dex2oat')
Orion Hodson4c3ade62021-02-10 14:07:10 +0000649 self._checker.check_executable("odrefresh")
Martin Stjernholm84a32cd2019-03-29 16:39:46 +0000650
Martin Stjernholm023c2182019-03-28 15:52:32 +0000651 # Check exported native libraries for Managed Core Library.
Victor Chang8b247622020-09-22 21:48:15 +0100652 self._checker.check_native_library('libicu')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000653 self._checker.check_native_library('libandroidio')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000654
655 # Check internal libraries for Managed Core Library.
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000656 self._checker.check_native_library('libexpat-host')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000657 self._checker.check_native_library('libz-host')
Andreas Gampe09123952019-01-30 13:17:02 -0800658
Luca Stefani4e91ee92019-03-06 15:08:16 +0100659
Andreas Gampe9dc4b052019-01-30 13:47:25 -0800660class DebugChecker:
661 def __init__(self, checker):
662 self._checker = checker
Luca Stefani4e91ee92019-03-06 15:08:16 +0100663
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000664 def __str__(self):
665 return 'Debug Checker'
666
667 def run(self):
Martin Stjernholm023c2182019-03-28 15:52:32 +0000668 # Check binaries for ART.
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000669 self._checker.check_executable('dexdiag')
Roland Levillain6dd62122019-06-18 19:10:21 +0100670 self._checker.check_executable('dexanalyze')
671 self._checker.check_executable('dexlayout')
672 self._checker.check_symlinked_multilib_executable('imgdiag')
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000673
Martin Stjernholm023c2182019-03-28 15:52:32 +0000674 # Check debug binaries for ART.
Roland Levillain6dd62122019-06-18 19:10:21 +0100675 self._checker.check_executable('dexlayoutd')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000676 self._checker.check_executable('dexoptanalyzerd')
Roland Levillain6dd62122019-06-18 19:10:21 +0100677 self._checker.check_symlinked_multilib_executable('imgdiagd')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000678 self._checker.check_executable('profmand')
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000679
Martin Stjernholm023c2182019-03-28 15:52:32 +0000680 # Check internal libraries for ART.
681 self._checker.check_native_library('libadbconnectiond')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000682 self._checker.check_native_library('libartbased')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000683 self._checker.check_native_library('libartd')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000684 self._checker.check_native_library('libartd-compiler')
685 self._checker.check_native_library('libartd-dexlayout')
Roland Levillain4d6f24d2019-10-02 16:47:36 +0100686 self._checker.check_native_library('libartd-disassembler')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000687 self._checker.check_native_library('libdexfiled')
688 self._checker.check_native_library('libopenjdkjvmd')
689 self._checker.check_native_library('libopenjdkjvmtid')
690 self._checker.check_native_library('libprofiled')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000691
692 # Check internal libraries for Managed Core Library.
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000693 self._checker.check_native_library('libopenjdkd')
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000694
Luca Stefani4e91ee92019-03-06 15:08:16 +0100695
Andreas Gampe9dc4b052019-01-30 13:47:25 -0800696class DebugTargetChecker:
697 def __init__(self, checker):
698 self._checker = checker
Luca Stefani4e91ee92019-03-06 15:08:16 +0100699
Andreas Gampea0242cf2019-01-29 13:01:23 -0800700 def __str__(self):
701 return 'Debug (Target) Checker'
702
703 def run(self):
Martin Stjernholm023c2182019-03-28 15:52:32 +0000704 # Check ART debug binaries.
Nicolas Geoffrayd68580e2020-04-15 16:52:32 +0100705 self._checker.check_multilib_executable('dex2oatd')
706 self._checker.check_multilib_executable('dex2oat')
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000707 self._checker.check_executable('oatdumpd')
Martin Stjernholm023c2182019-03-28 15:52:32 +0000708
709 # Check ART internal libraries.
Martin Stjernholm356864a2019-04-30 16:22:17 +0100710 self._checker.check_native_library('libdexfiled_external')
Florian Mayer07710c52019-09-16 15:53:38 +0000711 self._checker.check_native_library('libperfetto_hprofd')
Andreas Gampea0242cf2019-01-29 13:01:23 -0800712
Martin Stjernholm2babede2019-03-18 21:04:49 +0000713 # Check internal native library dependencies.
714 #
715 # Like in the release package, we check that we don't get other dependencies
716 # besides those listed here. In this case the concern is not bloat, but
717 # rather that we don't get behavioural differences between user (release)
718 # and userdebug/eng builds, which could happen if the debug package has
719 # duplicate library instances where releases don't. In other words, it's
720 # uncontroversial to add debug-only dependencies, as long as they don't make
721 # assumptions on having a single global state (ideally they should have
722 # double_loadable:true, cf. go/double_loadable). Also, like in the release
723 # package we need to look out for dependencies that should go through
724 # exported library stubs (until b/128708192 is fixed).
Martin Stjernholmc17ace22019-03-18 21:47:31 +0000725 self._checker.check_optional_native_library('libvixld') # Only on ARM/ARM64
Martin Stjernholm77f17662019-04-03 17:08:21 +0100726 self._checker.check_prefer64_library('libmeminfo')
727 self._checker.check_prefer64_library('libprocinfo')
Martin Stjernholm2babede2019-03-18 21:04:49 +0000728
729
Roland Levillain61f07162019-06-26 12:44:04 +0100730class TestingTargetChecker:
731 def __init__(self, checker):
732 self._checker = checker
733
734 def __str__(self):
735 return 'Testing (Target) Checker'
736
737 def run(self):
David Srbecky883c1342020-05-11 23:30:29 +0000738 # Check ART test binaries.
739 self._checker.check_art_test_executable('art_cmdline_tests')
740 self._checker.check_art_test_executable('art_compiler_tests')
741 self._checker.check_art_test_executable('art_dex2oat_tests')
742 self._checker.check_art_test_executable('art_dexanalyze_tests')
743 self._checker.check_art_test_executable('art_dexdiag_tests')
744 self._checker.check_art_test_executable('art_dexdump_tests')
745 self._checker.check_art_test_executable('art_dexlayout_tests')
746 self._checker.check_art_test_executable('art_dexlist_tests')
747 self._checker.check_art_test_executable('art_dexoptanalyzer_tests')
748 self._checker.check_art_test_executable('art_imgdiag_tests')
749 self._checker.check_art_test_executable('art_libartbase_tests')
750 self._checker.check_art_test_executable('art_libartpalette_tests')
Chris Wailesbfd622a2021-01-13 16:13:41 -0800751 self._checker.check_art_test_executable('art_libartservice_tests')
752 self._checker.check_art_test_executable('art_libarttools_tests')
Roland Levillain61f07162019-06-26 12:44:04 +0100753 self._checker.check_art_test_executable('art_libdexfile_support_tests')
David Srbecky883c1342020-05-11 23:30:29 +0000754 self._checker.check_art_test_executable('art_libdexfile_tests')
755 self._checker.check_art_test_executable('art_libprofile_tests')
756 self._checker.check_art_test_executable('art_oatdump_tests')
Orion Hodson4c3ade62021-02-10 14:07:10 +0000757 self._checker.check_art_test_executable('art_odrefresh_tests')
David Srbecky883c1342020-05-11 23:30:29 +0000758 self._checker.check_art_test_executable('art_profman_tests')
759 self._checker.check_art_test_executable('art_runtime_compiler_tests')
760 self._checker.check_art_test_executable('art_runtime_tests')
761 self._checker.check_art_test_executable('art_sigchain_tests')
Roland Levillain61f07162019-06-26 12:44:04 +0100762
763 # Check ART test (internal) libraries.
764 self._checker.check_native_library('libart-gtest')
Ulya Trafimovich5439f052020-07-29 10:03:46 +0100765 self._checker.check_native_library('libartd-simulator-container')
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')
Calin Juravle33787682019-07-26 14:27:18 -0700822 self._checker.check_art_test_data('art-gtest-jars-Dex2oatVdexTestDex.jar')
823 self._checker.check_art_test_data('art-gtest-jars-Dex2oatVdexPublicSdkDex.dex')
David Srbecky4a88a5a2020-05-05 16:21:57 +0100824
Roland Levillain61f07162019-06-26 12:44:04 +0100825
Martin Stjernholm2babede2019-03-18 21:04:49 +0000826class NoSuperfluousBinariesChecker:
827 def __init__(self, checker):
828 self._checker = checker
829
830 def __str__(self):
831 return 'No superfluous binaries checker'
832
833 def run(self):
834 self._checker.check_no_superfluous_files('bin')
835
836
837class NoSuperfluousLibrariesChecker:
838 def __init__(self, checker):
839 self._checker = checker
840
841 def __str__(self):
842 return 'No superfluous libraries checker'
843
844 def run(self):
845 self._checker.check_no_superfluous_files('javalib')
846 self._checker.check_no_superfluous_files('lib')
Martin Stjernholm2babede2019-03-18 21:04:49 +0000847 self._checker.check_no_superfluous_files('lib64')
Martin Stjernholm2babede2019-03-18 21:04:49 +0000848
Andreas Gampeb1d55672019-01-29 22:17:02 +0000849
Roland Levillain61f07162019-06-26 12:44:04 +0100850class NoSuperfluousArtTestsChecker:
851 def __init__(self, checker):
852 self._checker = checker
853
854 def __str__(self):
855 return 'No superfluous ART tests checker'
856
857 def run(self):
Orion Hodsonaf3ed622019-10-03 16:30:54 +0100858 for arch in ARCHS:
859 self._checker.check_no_superfluous_files('%s/%s' % (ART_TEST_DIR, arch))
Roland Levillain61f07162019-06-26 12:44:04 +0100860
861
Luca Stefani4e91ee92019-03-06 15:08:16 +0100862class List:
Roland Levillain77251f92019-08-09 15:00:04 +0100863 def __init__(self, provider, print_size=False):
Luca Stefani4e91ee92019-03-06 15:08:16 +0100864 self._provider = provider
Roland Levillaine91d7872019-07-25 18:56:06 +0100865 self._print_size = print_size
Luca Stefani4e91ee92019-03-06 15:08:16 +0100866
867 def print_list(self):
Roland Levillain09744ac2019-06-06 18:23:02 +0100868
869 def print_list_rec(path):
870 apex_map = self._provider.read_dir(path)
871 if apex_map is None:
872 return
873 apex_map = dict(apex_map)
874 if '.' in apex_map:
875 del apex_map['.']
876 if '..' in apex_map:
877 del apex_map['..']
878 for (_, val) in sorted(apex_map.items()):
879 val_path = os.path.join(path, val.name)
Roland Levillaine91d7872019-07-25 18:56:06 +0100880 if self._print_size:
881 if val.size < 0:
882 print('[ n/a ] %s' % val_path)
883 else:
884 print('[%11d] %s' % (val.size, val_path))
885 else:
886 print(val_path)
Roland Levillain09744ac2019-06-06 18:23:02 +0100887 if val.is_dir:
888 print_list_rec(val_path)
889
890 print_list_rec('')
Luca Stefani4e91ee92019-03-06 15:08:16 +0100891
892
893class Tree:
Roland Levillain77251f92019-08-09 15:00:04 +0100894 def __init__(self, provider, title, print_size=False):
Luca Stefani4e91ee92019-03-06 15:08:16 +0100895 print('%s' % title)
896 self._provider = provider
Luca Stefani4e91ee92019-03-06 15:08:16 +0100897 self._has_next_list = []
Roland Levillain88e55692019-07-25 15:57:06 +0100898 self._print_size = print_size
Luca Stefani4e91ee92019-03-06 15:08:16 +0100899
900 @staticmethod
901 def get_vertical(has_next_list):
902 string = ''
903 for v in has_next_list:
904 string += '%s ' % ('│' if v else ' ')
905 return string
906
907 @staticmethod
908 def get_last_vertical(last):
909 return '└── ' if last else '├── '
910
911 def print_tree(self):
Roland Levillain09744ac2019-06-06 18:23:02 +0100912
913 def print_tree_rec(path):
914 apex_map = self._provider.read_dir(path)
915 if apex_map is None:
916 return
917 apex_map = dict(apex_map)
918 if '.' in apex_map:
919 del apex_map['.']
920 if '..' in apex_map:
921 del apex_map['..']
922 key_list = list(sorted(apex_map.keys()))
923 for i, key in enumerate(key_list):
924 prev = self.get_vertical(self._has_next_list)
925 last = self.get_last_vertical(i == len(key_list) - 1)
926 val = apex_map[key]
Roland Levillain88e55692019-07-25 15:57:06 +0100927 if self._print_size:
928 if val.size < 0:
929 print('%s%s[ n/a ] %s' % (prev, last, val.name))
930 else:
931 print('%s%s[%11d] %s' % (prev, last, val.size, val.name))
932 else:
933 print('%s%s%s' % (prev, last, val.name))
Roland Levillain09744ac2019-06-06 18:23:02 +0100934 if val.is_dir:
935 self._has_next_list.append(i < len(key_list) - 1)
936 val_path = os.path.join(path, val.name)
937 print_tree_rec(val_path)
938 self._has_next_list.pop()
939
940 print_tree_rec('')
Luca Stefani4e91ee92019-03-06 15:08:16 +0100941
Andreas Gampeb1d55672019-01-29 22:17:02 +0000942
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000943# Note: do not sys.exit early, for __del__ cleanup.
Luca Stefani4e91ee92019-03-06 15:08:16 +0100944def art_apex_test_main(test_args):
Roland Levillaindd20d002019-07-19 16:09:47 +0100945 if test_args.host and test_args.flattened:
946 logging.error("Both of --host and --flattened set")
947 return 1
Luca Stefani4e91ee92019-03-06 15:08:16 +0100948 if test_args.list and test_args.tree:
Andreas Gampeb1d55672019-01-29 22:17:02 +0000949 logging.error("Both of --list and --tree set")
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000950 return 1
Roland Levillaine91d7872019-07-25 18:56:06 +0100951 if test_args.size and not (test_args.list or test_args.tree):
952 logging.error("--size set but neither --list nor --tree set")
Roland Levillain88e55692019-07-25 15:57:06 +0100953 return 1
Martin Stjernholm3cb59a42019-08-07 17:18:29 +0100954 if not test_args.flattened and not test_args.tmpdir:
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000955 logging.error("Need a tmpdir.")
956 return 1
Martin Stjernholm3cb59a42019-08-07 17:18:29 +0100957 if not test_args.flattened and not test_args.host and not test_args.debugfs:
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000958 logging.error("Need debugfs.")
959 return 1
960
Roland Levillainfdbe2072019-11-11 15:52:45 +0000961 if test_args.host:
962 # Host APEX.
963 if test_args.flavor not in [FLAVOR_DEBUG, FLAVOR_AUTO]:
964 logging.error("Using option --host with non-Debug APEX")
Roland Levillaindbd6a142019-10-03 15:08:28 +0100965 return 1
Roland Levillainfdbe2072019-11-11 15:52:45 +0000966 # Host APEX is always a debug flavor (for now).
967 test_args.flavor = FLAVOR_DEBUG
968 else:
969 # Device APEX.
970 if test_args.flavor == FLAVOR_AUTO:
971 logging.warning('--flavor=auto, trying to autodetect. This may be incorrect!')
972 for flavor in [ FLAVOR_RELEASE, FLAVOR_DEBUG, FLAVOR_TESTING ]:
973 flavor_pattern = '*.%s*' % flavor
974 if fnmatch.fnmatch(test_args.apex, flavor_pattern):
975 test_args.flavor = flavor
976 break
977 if test_args.flavor == FLAVOR_AUTO:
978 logging.error(' Could not detect APEX flavor, neither \'%s\', \'%s\' nor \'%s\' in \'%s\'',
979 FLAVOR_RELEASE, FLAVOR_DEBUG, FLAVOR_TESTING, test_args.apex)
980 return 1
Roland Levillaindbd6a142019-10-03 15:08:28 +0100981
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000982 try:
Luca Stefani4e91ee92019-03-06 15:08:16 +0100983 if test_args.host:
984 apex_provider = HostApexProvider(test_args.apex, test_args.tmpdir)
Andreas Gampe09123952019-01-30 13:17:02 -0800985 else:
Roland Levillaindd20d002019-07-19 16:09:47 +0100986 if test_args.flattened:
987 apex_provider = TargetFlattenedApexProvider(test_args.apex)
988 else:
989 apex_provider = TargetApexProvider(test_args.apex, test_args.tmpdir, test_args.debugfs)
Luca Stefani4e91ee92019-03-06 15:08:16 +0100990 except (zipfile.BadZipFile, zipfile.LargeZipFile) as e:
Andreas Gampea0242cf2019-01-29 13:01:23 -0800991 logging.error('Failed to create provider: %s', e)
Andreas Gampe2c846cd2019-01-29 22:06:46 +0000992 return 1
993
Luca Stefani4e91ee92019-03-06 15:08:16 +0100994 if test_args.tree:
Roland Levillain88e55692019-07-25 15:57:06 +0100995 Tree(apex_provider, test_args.apex, test_args.size).print_tree()
Andreas Gampeb1d55672019-01-29 22:17:02 +0000996 return 0
Luca Stefani4e91ee92019-03-06 15:08:16 +0100997 if test_args.list:
Roland Levillaine91d7872019-07-25 18:56:06 +0100998 List(apex_provider, test_args.size).print_list()
Andreas Gampeb1d55672019-01-29 22:17:02 +0000999 return 0
1000
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001001 checkers = []
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001002 if test_args.bitness == BITNESS_AUTO:
Luca Stefani4e91ee92019-03-06 15:08:16 +01001003 logging.warning('--bitness=auto, trying to autodetect. This may be incorrect!')
Andreas Gampeeb555b82019-01-30 14:47:49 -08001004 has_32 = apex_provider.get('lib') is not None
1005 has_64 = apex_provider.get('lib64') is not None
1006 if has_32 and has_64:
Luca Stefani4e91ee92019-03-06 15:08:16 +01001007 logging.warning(' Detected multilib')
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001008 test_args.bitness = BITNESS_MULTILIB
Andreas Gampeeb555b82019-01-30 14:47:49 -08001009 elif has_32:
Luca Stefani4e91ee92019-03-06 15:08:16 +01001010 logging.warning(' Detected 32-only')
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001011 test_args.bitness = BITNESS_32
Andreas Gampeeb555b82019-01-30 14:47:49 -08001012 elif has_64:
Luca Stefani4e91ee92019-03-06 15:08:16 +01001013 logging.warning(' Detected 64-only')
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001014 test_args.bitness = BITNESS_64
Andreas Gampeeb555b82019-01-30 14:47:49 -08001015 else:
1016 logging.error(' Could not detect bitness, neither lib nor lib64 contained.')
Roland Levillain77251f92019-08-09 15:00:04 +01001017 List(apex_provider).print_list()
Andreas Gampeeb555b82019-01-30 14:47:49 -08001018 return 1
1019
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001020 if test_args.bitness == BITNESS_32:
Andreas Gampeeb555b82019-01-30 14:47:49 -08001021 base_checker = Arch32Checker(apex_provider)
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001022 elif test_args.bitness == BITNESS_64:
Andreas Gampeeb555b82019-01-30 14:47:49 -08001023 base_checker = Arch64Checker(apex_provider)
1024 else:
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001025 assert test_args.bitness == BITNESS_MULTILIB
Andreas Gampeeb555b82019-01-30 14:47:49 -08001026 base_checker = MultilibChecker(apex_provider)
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001027
Andreas Gampe9dc4b052019-01-30 13:47:25 -08001028 checkers.append(ReleaseChecker(base_checker))
Luca Stefani4e91ee92019-03-06 15:08:16 +01001029 if test_args.host:
Andreas Gampe9dc4b052019-01-30 13:47:25 -08001030 checkers.append(ReleaseHostChecker(base_checker))
Andreas Gampe09123952019-01-30 13:17:02 -08001031 else:
Andreas Gampe9dc4b052019-01-30 13:47:25 -08001032 checkers.append(ReleaseTargetChecker(base_checker))
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001033 if test_args.flavor == FLAVOR_DEBUG or test_args.flavor == FLAVOR_TESTING:
Andreas Gampe9dc4b052019-01-30 13:47:25 -08001034 checkers.append(DebugChecker(base_checker))
Roland Levillain61f07162019-06-26 12:44:04 +01001035 if not test_args.host:
1036 checkers.append(DebugTargetChecker(base_checker))
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001037 if test_args.flavor == FLAVOR_TESTING:
Roland Levillain61f07162019-06-26 12:44:04 +01001038 checkers.append(TestingTargetChecker(base_checker))
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001039
Martin Stjernholm2babede2019-03-18 21:04:49 +00001040 # These checkers must be last.
1041 checkers.append(NoSuperfluousBinariesChecker(base_checker))
Roland Levillain61f07162019-06-26 12:44:04 +01001042 checkers.append(NoSuperfluousArtTestsChecker(base_checker))
Martin Stjernholm2babede2019-03-18 21:04:49 +00001043 if not test_args.host:
1044 # We only care about superfluous libraries on target, where their absence
1045 # can be vital to ensure they get picked up from the right package.
1046 checkers.append(NoSuperfluousLibrariesChecker(base_checker))
1047
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001048 failed = False
1049 for checker in checkers:
1050 logging.info('%s...', checker)
1051 checker.run()
Andreas Gampe9dc4b052019-01-30 13:47:25 -08001052 if base_checker.error_count() > 0:
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001053 logging.error('%s FAILED', checker)
1054 failed = True
1055 else:
1056 logging.info('%s SUCCEEDED', checker)
Andreas Gampe9dc4b052019-01-30 13:47:25 -08001057 base_checker.reset_errors()
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001058
1059 return 1 if failed else 0
1060
Luca Stefani4e91ee92019-03-06 15:08:16 +01001061
1062def art_apex_test_default(test_parser):
1063 if 'ANDROID_PRODUCT_OUT' not in os.environ:
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001064 logging.error('No-argument use requires ANDROID_PRODUCT_OUT')
1065 sys.exit(1)
1066 product_out = os.environ['ANDROID_PRODUCT_OUT']
Luca Stefani4e91ee92019-03-06 15:08:16 +01001067 if 'ANDROID_HOST_OUT' not in os.environ:
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001068 logging.error('No-argument use requires ANDROID_HOST_OUT')
1069 sys.exit(1)
1070 host_out = os.environ['ANDROID_HOST_OUT']
1071
Roland Levillain4ee567f2020-07-27 17:02:54 +01001072 test_args = test_parser.parse_args(['unused']) # For consistency.
Luca Stefani4e91ee92019-03-06 15:08:16 +01001073 test_args.debugfs = '%s/bin/debugfs' % host_out
1074 test_args.tmpdir = '.'
1075 test_args.tree = False
1076 test_args.list = False
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001077 test_args.bitness = BITNESS_AUTO
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001078 failed = False
1079
Luca Stefani4e91ee92019-03-06 15:08:16 +01001080 if not os.path.exists(test_args.debugfs):
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001081 logging.error("Cannot find debugfs (default path %s). Please build it, e.g., m debugfs",
Luca Stefani4e91ee92019-03-06 15:08:16 +01001082 test_args.debugfs)
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001083 sys.exit(1)
1084
Roland Levillaind7ea0432019-09-09 16:29:12 +01001085 # TODO: Add host support.
1086 # TODO: Add support for flattened APEX packages.
Luca Stefani4e91ee92019-03-06 15:08:16 +01001087 configs = [
Martin Stjernholm3e9abfc2020-10-09 22:17:51 +01001088 {'name': 'com.android.art', 'flavor': FLAVOR_RELEASE, 'host': False},
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001089 {'name': 'com.android.art.debug', 'flavor': FLAVOR_DEBUG, 'host': False},
1090 {'name': 'com.android.art.testing', 'flavor': FLAVOR_TESTING, 'host': False},
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001091 ]
1092
1093 for config in configs:
1094 logging.info(config['name'])
1095 # TODO: Host will need different path.
Luca Stefani4e91ee92019-03-06 15:08:16 +01001096 test_args.apex = '%s/system/apex/%s.apex' % (product_out, config['name'])
1097 if not os.path.exists(test_args.apex):
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001098 failed = True
Luca Stefani4e91ee92019-03-06 15:08:16 +01001099 logging.error("Cannot find APEX %s. Please build it first.", test_args.apex)
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001100 continue
Roland Levillaind7ea0432019-09-09 16:29:12 +01001101 test_args.flavor = config['flavor']
Luca Stefani4e91ee92019-03-06 15:08:16 +01001102 test_args.host = config['host']
1103 failed = art_apex_test_main(test_args) != 0
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001104
1105 if failed:
1106 sys.exit(1)
1107
Luca Stefani4e91ee92019-03-06 15:08:16 +01001108
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001109if __name__ == "__main__":
Martin Stjernholme58624f2019-09-20 15:53:40 +01001110 parser = argparse.ArgumentParser(description='Check integrity of an ART APEX.')
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001111
Roland Levillaind7ea0432019-09-09 16:29:12 +01001112 parser.add_argument('apex', help='APEX file input')
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001113
Roland Levillaind7ea0432019-09-09 16:29:12 +01001114 parser.add_argument('--host', help='Check as host APEX', action='store_true')
Andreas Gampe09123952019-01-30 13:17:02 -08001115
Roland Levillaind7ea0432019-09-09 16:29:12 +01001116 parser.add_argument('--flattened', help='Check as flattened (target) APEX', action='store_true')
Roland Levillaindd20d002019-07-19 16:09:47 +01001117
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001118 parser.add_argument('--flavor', help='Check as FLAVOR APEX', choices=FLAVORS_ALL,
1119 default=FLAVOR_AUTO)
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001120
Andreas Gampeb1d55672019-01-29 22:17:02 +00001121 parser.add_argument('--list', help='List all files', action='store_true')
1122 parser.add_argument('--tree', help='Print directory tree', action='store_true')
Roland Levillaine91d7872019-07-25 18:56:06 +01001123 parser.add_argument('--size', help='Print file sizes', action='store_true')
Andreas Gampeb1d55672019-01-29 22:17:02 +00001124
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001125 parser.add_argument('--tmpdir', help='Directory for temp files')
1126 parser.add_argument('--debugfs', help='Path to debugfs')
1127
Orion Hodsonaf3ed622019-10-03 16:30:54 +01001128 parser.add_argument('--bitness', help='Bitness to check', choices=BITNESS_ALL,
1129 default=BITNESS_AUTO)
Andreas Gampeeb555b82019-01-30 14:47:49 -08001130
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001131 if len(sys.argv) == 1:
Luca Stefani4e91ee92019-03-06 15:08:16 +01001132 art_apex_test_default(parser)
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001133 else:
1134 args = parser.parse_args()
1135
1136 if args is None:
1137 sys.exit(1)
1138
Luca Stefani4e91ee92019-03-06 15:08:16 +01001139 exit_code = art_apex_test_main(args)
Andreas Gampe2c846cd2019-01-29 22:06:46 +00001140 sys.exit(exit_code)