blob: 5dcc76f243564bf95f9995ad298578654084506c [file] [log] [blame]
Alan Viverette7d87cea2017-09-26 13:39:21 -04001#!/usr/bin/python3
2#
3# Copyright (C) 2015 The Android Open Source Project
4#
5# Licensed under the Apache License, Version 2.0 (the 'License');
6# you may not use this file except in compliance with the License.
7# You may obtain a copy of the License at
8#
9# http://www.apache.org/licenses/LICENSE-2.0
10#
11# Unless required by applicable law or agreed to in writing, software
12# distributed under the License is distributed on an 'AS IS' BASIS,
13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14# See the License for the specific language governing permissions and
15# limitations under the License.
16#
17
18import os
19import sys
20import tempfile
21import threading
22import time
23import traceback
24
25from android_device import *
26from avd import *
27from queue import Queue, Empty
28
29
Alan Viverette377b3e72017-10-03 13:06:45 -040030# This dict should contain one entry for every density listed in CDD 7.1.1.3.
Alan Viverette7d87cea2017-09-26 13:39:21 -040031CTS_THEME_dict = {
32 120: "ldpi",
33 160: "mdpi",
34 213: "tvdpi",
35 240: "hdpi",
Alan Viverette377b3e72017-10-03 13:06:45 -040036 260: "260dpi",
37 280: "280dpi",
38 300: "300dpi",
Alan Viverette7d87cea2017-09-26 13:39:21 -040039 320: "xhdpi",
Alan Viverette377b3e72017-10-03 13:06:45 -040040 340: "340dpi",
Alan Viverette7d87cea2017-09-26 13:39:21 -040041 360: "360dpi",
Alan Viverette377b3e72017-10-03 13:06:45 -040042 400: "400dpi",
Alan Viverette7d87cea2017-09-26 13:39:21 -040043 420: "420dpi",
44 480: "xxhdpi",
45 560: "560dpi",
46 640: "xxxhdpi",
47}
48
49OUT_FILE = "/sdcard/cts-theme-assets.zip"
50
51
52class ParallelExecutor(threading.Thread):
Alan Viveretteb0e0b5e2017-11-06 10:24:24 -050053 def __init__(self, tasks, setup, q):
Alan Viverette7d87cea2017-09-26 13:39:21 -040054 threading.Thread.__init__(self)
55 self._q = q
56 self._tasks = tasks
57 self._setup = setup
58 self._result = 0
59
60 def run(self):
61 try:
62 while True:
Alan Viveretteb0e0b5e2017-11-06 10:24:24 -050063 config = self._q.get(block=True, timeout=2)
Alan Viverette7d87cea2017-09-26 13:39:21 -040064 for t in self._tasks:
65 try:
66 if t(self._setup, config):
67 self._result += 1
68 except KeyboardInterrupt:
69 raise
70 except:
71 print("Failed to execute thread:", sys.exc_info()[0])
72 traceback.print_exc()
Alan Viveretteb0e0b5e2017-11-06 10:24:24 -050073 self._q.task_done()
Alan Viverette7d87cea2017-09-26 13:39:21 -040074 except KeyboardInterrupt:
75 raise
76 except Empty:
77 pass
78
79 def get_result(self):
80 return self._result
81
82
83# pass a function with number of instances to be executed in parallel
84# each thread continues until config q is empty.
85def execute_parallel(tasks, setup, q, num_threads):
86 result = 0
87 threads = []
88 for i in range(num_threads):
Alan Viveretteb0e0b5e2017-11-06 10:24:24 -050089 t = ParallelExecutor(tasks, setup, q)
Alan Viverette7d87cea2017-09-26 13:39:21 -040090 t.start()
91 threads.append(t)
92 for t in threads:
93 t.join()
94 result += t.get_result()
95 return result
96
97
98def print_adb_result(device, out, err):
99 print("device: " + device)
100 if out is not None:
101 print("out:\n" + out)
102 if err is not None:
103 print("err:\n" + err)
104
105
106def do_capture(setup, device_serial):
107 (themeApkPath, out_path) = setup
108
109 device = AndroidDevice(device_serial)
110
111 version = device.get_version_codename()
112 if version == "REL":
113 version = str(device.get_version_sdk())
114
115 density = device.get_density()
116
117 if CTS_THEME_dict[density]:
118 density_bucket = CTS_THEME_dict[density]
119 else:
120 density_bucket = str(density) + "dpi"
121
122 out_file = os.path.join(out_path, os.path.join(version, "%s.zip" % density_bucket))
123
124 device.uninstall_package('android.theme.app')
125
126 (out, err, success) = device.install_apk(themeApkPath)
127 if not success:
128 print("Failed to install APK on " + device_serial)
129 print_adb_result(device_serial, out, err)
130 return False
131
132 print("Generating images on " + device_serial + "...")
133 try:
134 (out, err) = device.run_instrumentation_test(
135 "android.theme.app/android.support.test.runner.AndroidJUnitRunner")
136 except KeyboardInterrupt:
137 raise
138 except:
139 (out, err) = device.run_instrumentation_test(
140 "android.theme.app/android.test.InstrumentationTestRunner")
141
142 # Detect test failure and abort.
143 if "FAILURES!!!" in out.split():
144 print_adb_result(device_serial, out, err)
145 return False
146
147 # Make sure that the run is complete by checking the process itself
148 print("Waiting for " + device_serial + "...")
149 wait_time = 0
150 while device.is_process_alive("android.theme.app"):
151 time.sleep(1)
152 wait_time = wait_time + 1
153 if wait_time > 180:
154 print("Timed out")
155 break
156
157 time.sleep(10)
158
159 print("Pulling images from " + device_serial + " to " + out_file)
160 device.run_adb_command("pull " + OUT_FILE + " " + out_file)
161 device.run_adb_command("shell rm -rf " + OUT_FILE)
162 return True
163
164
165def get_emulator_path():
166 if 'ANDROID_SDK_ROOT' not in os.environ:
167 print('Environment variable ANDROID_SDK_ROOT must point to your Android SDK root.')
168 sys.exit(1)
169
170 sdk_path = os.environ['ANDROID_SDK_ROOT']
171 if not os.path.isdir(sdk_path):
172 print("Failed to find Android SDK at ANDROID_SDK_ROOT: %s" % sdk_path)
173 sys.exit(1)
174
175 emu_path = os.path.join(os.path.join(sdk_path, 'tools'), 'emulator')
176 if not os.path.isfile(emu_path):
177 print("Failed to find emulator within ANDROID_SDK_ROOT: %s" % sdk_path)
178 sys.exit(1)
179
180 return emu_path
181
182
183def start_emulator(name, density):
Stan Iliev3baab432017-11-10 11:28:48 -0500184 if name == "local":
185 emu_path = ""
186 else:
187 emu_path = get_emulator_path()
Alan Viverette7d87cea2017-09-26 13:39:21 -0400188
189 # Start emulator for 560dpi, normal screen size.
190 test_avd = AVD(name, emu_path)
191 test_avd.configure_screen(density, 360, 640)
192 test_avd.start()
193 try:
194 test_avd_device = test_avd.get_device()
195 test_avd_device.wait_for_device()
196 test_avd_device.wait_for_boot_complete()
197 return test_avd
198 except:
199 test_avd.stop()
200 return None
201
202
203def main(argv):
204 if 'ANDROID_BUILD_TOP' not in os.environ or 'ANDROID_HOST_OUT' not in os.environ:
205 print('Missing environment variables. Did you run build/envsetup.sh and lunch?')
206 sys.exit(1)
207
208 theme_apk = os.path.join(os.environ['ANDROID_HOST_OUT'],
209 'cts/android-cts/testcases/CtsThemeDeviceApp.apk')
210 if not os.path.isfile(theme_apk):
211 print('Couldn\'t find test APK. Did you run make cts?')
212 sys.exit(1)
213
214 out_path = os.path.join(os.environ['ANDROID_BUILD_TOP'],
215 'cts/hostsidetests/theme/assets')
216 os.system("mkdir -p %s" % out_path)
217
218 if len(argv) is 2:
219 for density in CTS_THEME_dict.keys():
220 emulator = start_emulator(argv[1], density)
221 result = do_capture(setup=(theme_apk, out_path), device_serial=emulator.get_serial())
222 emulator.stop()
223 if result:
224 print("Generated reference images for %ddpi" % density)
225 else:
226 print("Failed to generate reference images for %ddpi" % density)
227 break
228 else:
229 tasks = [do_capture]
230 setup = (theme_apk, out_path)
231
Alan Viveretteb0e0b5e2017-11-06 10:24:24 -0500232 devices = enumerate_android_devices()
Alan Viverette7d87cea2017-09-26 13:39:21 -0400233
Alan Viveretteb0e0b5e2017-11-06 10:24:24 -0500234 if len(devices) > 0:
235 device_queue = Queue()
236 for device in devices:
237 device_queue.put(device)
Alan Viverette7d87cea2017-09-26 13:39:21 -0400238
Alan Viveretteb0e0b5e2017-11-06 10:24:24 -0500239 result = execute_parallel(tasks, setup, device_queue, len(devices))
Alan Viverette7d87cea2017-09-26 13:39:21 -0400240
Alan Viveretteb0e0b5e2017-11-06 10:24:24 -0500241 if result > 0:
242 print('Generated reference images for %(count)d devices' % {"count": result})
243 else:
244 print('Failed to generate reference images')
Alan Viverette7d87cea2017-09-26 13:39:21 -0400245 else:
Alan Viveretteb0e0b5e2017-11-06 10:24:24 -0500246 print('No devices found')
Alan Viverette7d87cea2017-09-26 13:39:21 -0400247
248
249if __name__ == '__main__':
250 main(sys.argv)