1 /*
2  * Copyright (C) 2018 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.server.wm.activity;
18 
19 import static androidx.test.InstrumentationRegistry.getInstrumentation;
20 
21 import static org.junit.Assert.assertArrayEquals;
22 import static org.junit.Assert.assertEquals;
23 
24 import android.app.ActivityManager;
25 import android.app.KeyguardManager;
26 import android.content.Context;
27 import android.content.nano.DeviceConfigurationProto;
28 import android.content.nano.GlobalConfigurationProto;
29 import android.content.nano.ResourcesConfigurationProto;
30 import android.content.pm.ConfigurationInfo;
31 import android.content.pm.FeatureInfo;
32 import android.content.pm.PackageManager;
33 import android.content.pm.SharedLibraryInfo;
34 import android.content.res.Configuration;
35 import android.hardware.display.DisplayManager;
36 import android.opengl.GLES10;
37 import android.os.Build;
38 import android.os.LocaleList;
39 import android.os.ParcelFileDescriptor;
40 import android.platform.test.annotations.Presubmit;
41 import android.server.wm.VirtualDisplayHelper;
42 import android.text.TextUtils;
43 import android.util.DisplayMetrics;
44 import android.view.Display;
45 
46 import com.google.protobuf.nano.InvalidProtocolBufferNanoException;
47 
48 import org.junit.Before;
49 import org.junit.Test;
50 
51 import java.io.ByteArrayOutputStream;
52 import java.io.FileInputStream;
53 import java.io.IOException;
54 import java.nio.charset.StandardCharsets;
55 import java.util.Arrays;
56 import java.util.Collections;
57 import java.util.Comparator;
58 import java.util.HashSet;
59 import java.util.List;
60 import java.util.Set;
61 
62 import javax.microedition.khronos.egl.EGL10;
63 import javax.microedition.khronos.egl.EGLConfig;
64 import javax.microedition.khronos.egl.EGLContext;
65 import javax.microedition.khronos.egl.EGLDisplay;
66 import javax.microedition.khronos.egl.EGLSurface;
67 
68 @Presubmit
69 public class ActivityManagerGetConfigTests {
70     Context mContext;
71     ActivityManager mAm;
72     PackageManager mPm;
73 
74     @Before
setUp()75     public void setUp() throws Exception {
76         mContext = getInstrumentation().getTargetContext();
77         mAm = mContext.getSystemService(ActivityManager.class);
78         mPm = mContext.getPackageManager();
79     }
80 
executeShellCommand(String cmd)81     private byte[] executeShellCommand(String cmd) {
82         try {
83             ParcelFileDescriptor pfd = getInstrumentation().getUiAutomation()
84                     .executeShellCommand(cmd);
85             byte[] buf = new byte[512];
86             int bytesRead;
87             FileInputStream fis = new ParcelFileDescriptor.AutoCloseInputStream(pfd);
88             ByteArrayOutputStream stdout = new ByteArrayOutputStream();
89             while ((bytesRead = fis.read(buf)) != -1) {
90                 stdout.write(buf, 0, bytesRead);
91             }
92             fis.close();
93             return stdout.toByteArray();
94         } catch (IOException e) {
95             throw new RuntimeException(e);
96         }
97     }
98 
99     /**
100      * Adds all supported GL extensions for a provided EGLConfig to a set by creating an EGLContext
101      * and EGLSurface and querying extensions.
102      *
103      * @param egl An EGL API object
104      * @param display An EGLDisplay to create a context and surface with
105      * @param config The EGLConfig to get the extensions for
106      * @param surfaceSize eglCreatePbufferSurface generic parameters
107      * @param contextAttribs eglCreateContext generic parameters
108      * @param glExtensions A Set<String> to add GL extensions to
109      */
addExtensionsForConfig( EGL10 egl, EGLDisplay display, EGLConfig config, int[] surfaceSize, int[] contextAttribs, Set<String> glExtensions)110     private static void addExtensionsForConfig(
111             EGL10 egl,
112             EGLDisplay display,
113             EGLConfig config,
114             int[] surfaceSize,
115             int[] contextAttribs,
116             Set<String> glExtensions) {
117         // Create a context.
118         EGLContext context =
119                 egl.eglCreateContext(display, config, EGL10.EGL_NO_CONTEXT, contextAttribs);
120         // No-op if we can't create a context.
121         if (context == EGL10.EGL_NO_CONTEXT) {
122             return;
123         }
124 
125         // Create a surface.
126         EGLSurface surface = egl.eglCreatePbufferSurface(display, config, surfaceSize);
127         if (surface == EGL10.EGL_NO_SURFACE) {
128             egl.eglDestroyContext(display, context);
129             return;
130         }
131 
132         // Update the current surface and context.
133         egl.eglMakeCurrent(display, surface, surface, context);
134 
135         // Get the list of extensions.
136         String extensionList = GLES10.glGetString(GLES10.GL_EXTENSIONS);
137         if (!TextUtils.isEmpty(extensionList)) {
138             // The list of extensions comes from the driver separated by spaces.
139             // Split them apart and add them into a Set for deduping purposes.
140             for (String extension : extensionList.split(" ")) {
141                 glExtensions.add(extension);
142             }
143         }
144 
145         // Tear down the context and surface for this config.
146         egl.eglMakeCurrent(display, EGL10.EGL_NO_SURFACE, EGL10.EGL_NO_SURFACE, EGL10.EGL_NO_CONTEXT);
147         egl.eglDestroySurface(display, surface);
148         egl.eglDestroyContext(display, context);
149     }
150 
151 
getGlExtensionsFromDriver()152     Set<String> getGlExtensionsFromDriver() {
153         Set<String> glExtensions = new HashSet<>();
154 
155         // Get the EGL implementation.
156         EGL10 egl = (EGL10) EGLContext.getEGL();
157         if (egl == null) {
158             throw new RuntimeException("Warning: couldn't get EGL");
159         }
160 
161         // Get the default display and initialize it.
162         EGLDisplay display = egl.eglGetDisplay(EGL10.EGL_DEFAULT_DISPLAY);
163         int[] version = new int[2];
164         egl.eglInitialize(display, version);
165 
166         // Call getConfigs() in order to find out how many there are.
167         int[] numConfigs = new int[1];
168         if (!egl.eglGetConfigs(display, null, 0, numConfigs)) {
169             throw new RuntimeException("Warning: couldn't get EGL config count");
170         }
171 
172         // Allocate space for all configs and ask again.
173         EGLConfig[] configs = new EGLConfig[numConfigs[0]];
174         if (!egl.eglGetConfigs(display, configs, numConfigs[0], numConfigs)) {
175             throw new RuntimeException("Warning: couldn't get EGL configs");
176         }
177 
178         // Allocate surface size parameters outside of the main loop to cut down
179         // on GC thrashing.  1x1 is enough since we are only using it to get at
180         // the list of extensions.
181         int[] surfaceSize =
182                 new int[] {
183                         EGL10.EGL_WIDTH, 1,
184                         EGL10.EGL_HEIGHT, 1,
185                         EGL10.EGL_NONE
186                 };
187 
188         // For when we need to create a GLES2.0 context.
189         final int EGL_CONTEXT_CLIENT_VERSION = 0x3098;
190         int[] gles2 = new int[] {EGL_CONTEXT_CLIENT_VERSION, 2, EGL10.EGL_NONE};
191 
192         // For getting return values from eglGetConfigAttrib
193         int[] attrib = new int[1];
194 
195         for (int i = 0; i < numConfigs[0]; i++) {
196             // Get caveat for this config in order to skip slow (i.e. software) configs.
197             egl.eglGetConfigAttrib(display, configs[i], EGL10.EGL_CONFIG_CAVEAT, attrib);
198             if (attrib[0] == EGL10.EGL_SLOW_CONFIG) {
199                 continue;
200             }
201 
202             // If the config does not support pbuffers we cannot do an eglMakeCurrent
203             // on it in addExtensionsForConfig(), so skip it here. Attempting to make
204             // it current with a pbuffer will result in an EGL_BAD_MATCH error
205             egl.eglGetConfigAttrib(display, configs[i], EGL10.EGL_SURFACE_TYPE, attrib);
206             if ((attrib[0] & EGL10.EGL_PBUFFER_BIT) == 0) {
207                 continue;
208             }
209 
210             final int EGL_OPENGL_ES_BIT = 0x0001;
211             final int EGL_OPENGL_ES2_BIT = 0x0004;
212             egl.eglGetConfigAttrib(display, configs[i], EGL10.EGL_RENDERABLE_TYPE, attrib);
213             if ((attrib[0] & EGL_OPENGL_ES_BIT) != 0) {
214                 addExtensionsForConfig(egl, display, configs[i], surfaceSize, null, glExtensions);
215             }
216             if ((attrib[0] & EGL_OPENGL_ES2_BIT) != 0) {
217                 addExtensionsForConfig(egl, display, configs[i], surfaceSize, gles2, glExtensions);
218             }
219         }
220 
221         // Release all EGL resources.
222         egl.eglTerminate(display);
223 
224         return glExtensions;
225     }
226 
checkResourceConfig(Configuration config, DisplayMetrics metrics, ResourcesConfigurationProto resConfig)227     private void checkResourceConfig(Configuration config, DisplayMetrics metrics,
228             ResourcesConfigurationProto resConfig) {
229         final int width, height;
230         if (metrics.widthPixels >= metrics.heightPixels) {
231             width = metrics.widthPixels;
232             height = metrics.heightPixels;
233         } else {
234             //noinspection SuspiciousNameCombination
235             width = metrics.heightPixels;
236             //noinspection SuspiciousNameCombination
237             height = metrics.widthPixels;
238         }
239 
240         assertEquals("Expected SDK version does not match",
241                 Build.VERSION.RESOURCES_SDK_INT, resConfig.sdkVersion);
242         assertEquals("Expected screen width px does not match",
243                 width, resConfig.screenWidthPx);
244         assertEquals("Expected screen width px does not match",
245                 height, resConfig.screenHeightPx);
246 
247         assertEquals("Expected font scale does not match",
248                 config.fontScale, resConfig.configuration.fontScale, Float.MIN_VALUE*5);
249         assertEquals("Expected mcc does not match",
250                 config.mcc, resConfig.configuration.mcc);
251         assertEquals("Expected mnc does not match",
252                 config.mnc, resConfig.configuration.mnc);
253         LocaleList llist = config.getLocales();
254         LocaleList lprotos = LocaleList.forLanguageTags(resConfig.configuration.localeList);
255         assertEquals("Expected number of locales does not match", llist.size(), lprotos.size());
256         for (int i = 0; i < llist.size(); i++) {
257             assertEquals("Expected locale #" + i + " does not match",
258                     llist.get(i).toLanguageTag(), lprotos.get(i).toLanguageTag());
259         }
260         assertEquals("Expected screen layout does not match",
261                 config.screenLayout, resConfig.configuration.screenLayout);
262         assertEquals("Expected color mode does not match",
263                 config.colorMode, resConfig.configuration.colorMode);
264         assertEquals("Expected touchscreen does not match",
265                 config.touchscreen, resConfig.configuration.touchscreen);
266         assertEquals("Expected keyboard does not match",
267                 config.keyboard, resConfig.configuration.keyboard);
268         assertEquals("Expected keyboard hidden does not match",
269                 config.keyboardHidden, resConfig.configuration.keyboardHidden);
270         assertEquals("Expected hard keyboard hidden does not match",
271                 config.hardKeyboardHidden, resConfig.configuration.hardKeyboardHidden);
272         assertEquals("Expected navigation does not match",
273                 config.navigation, resConfig.configuration.navigation);
274         assertEquals("Expected navigation hidden does not match",
275                 config.navigationHidden, resConfig.configuration.navigationHidden);
276         assertEquals("Expected orientation does not match",
277                 config.orientation, resConfig.configuration.orientation);
278         assertEquals("Expected UI mode does not match",
279                 config.uiMode, resConfig.configuration.uiMode);
280         assertEquals("Expected screen width dp does not match",
281                 config.screenWidthDp, resConfig.configuration.screenWidthDp);
282         assertEquals("Expected screen hight dp does not match",
283                 config.screenHeightDp, resConfig.configuration.screenHeightDp);
284         assertEquals("Expected smallest screen width dp does not match",
285                 config.smallestScreenWidthDp, resConfig.configuration.smallestScreenWidthDp);
286         assertEquals("Expected density dpi does not match",
287                 config.densityDpi, resConfig.configuration.densityDpi);
288         // XXX not comparing windowConfiguration, since by definition this is contextual.
289     }
290 
checkDeviceConfig(DisplayMetrics displayMetrics, DeviceConfigurationProto deviceConfig)291     private void checkDeviceConfig(DisplayMetrics displayMetrics,
292             DeviceConfigurationProto deviceConfig) {
293         assertEquals("Expected stable screen width does not match",
294                 displayMetrics.widthPixels, deviceConfig.stableScreenWidthPx);
295         assertEquals("Expected stable screen height does not match",
296                 displayMetrics.heightPixels, deviceConfig.stableScreenHeightPx);
297         assertEquals("Expected stable screen density does not match",
298                 DisplayMetrics.DENSITY_DEVICE_STABLE, deviceConfig.stableDensityDpi);
299 
300         assertEquals("Expected total RAM does not match",
301                 mAm.getTotalRam(), deviceConfig.totalRam);
302         assertEquals("Expected low RAM does not match",
303                 mAm.isLowRamDevice(), deviceConfig.lowRam);
304         assertEquals("Expected max cores does not match",
305                 Runtime.getRuntime().availableProcessors(), deviceConfig.maxCores);
306         KeyguardManager kgm = mContext.getSystemService(KeyguardManager.class);
307         assertEquals("Expected has secure screen lock does not match",
308                 kgm.isDeviceSecure(), deviceConfig.hasSecureScreenLock);
309 
310         ConfigurationInfo configInfo = mAm.getDeviceConfigurationInfo();
311         if (configInfo.reqGlEsVersion != ConfigurationInfo.GL_ES_VERSION_UNDEFINED) {
312             assertEquals("Expected opengl version does not match",
313                     configInfo.reqGlEsVersion, deviceConfig.openglVersion);
314         }
315 
316         Set<String> glExtensionsSet = getGlExtensionsFromDriver();
317         String[] glExtensions = new String[glExtensionsSet.size()];
318         glExtensions = glExtensionsSet.toArray(glExtensions);
319         Arrays.sort(glExtensions);
320         assertArrayEquals("Expected opengl extensions does not match",
321                 glExtensions, deviceConfig.openglExtensions);
322 
323         List<SharedLibraryInfo> slibs = mPm.getSharedLibraries(0);
324         Collections.sort(slibs, Comparator.comparing(SharedLibraryInfo::getName));
325         String[] slibNames = new String[slibs.size()];
326         for (int i = 0; i < slibs.size(); i++) {
327             slibNames[i] = slibs.get(i).getName();
328         }
329         assertArrayEquals("Expected shared libraries does not match",
330                 slibNames, deviceConfig.sharedLibraries);
331 
332         FeatureInfo[] features = mPm.getSystemAvailableFeatures();
333         Arrays.sort(features, (o1, o2) -> {
334             if (o1.name == o2.name) return 0;
335             if (o1.name == null) return -1;
336             if (o2.name == null) return 1;
337             return o1.name.compareTo(o2.name);
338         });
339 
340         int size = 0;
341         for (int i = 0; i < features.length; i++) {
342             if (features[i].name != null) {
343                 size++;
344             }
345         }
346         String[] featureNames = new String[size];
347         for (int i = 0, j = 0; i < features.length; i++) {
348             if (features[i].name != null) {
349                 featureNames[j] = features[i].name;
350                 j++;
351             }
352         }
353         assertArrayEquals("Expected features does not match",
354                 featureNames, deviceConfig.features);
355     }
356 
357     @Test
testDeviceConfig()358     public void testDeviceConfig() {
359         byte[] dump = executeShellCommand("cmd activity get-config --proto --device");
360         GlobalConfigurationProto globalConfig;
361         try {
362             globalConfig = GlobalConfigurationProto.parseFrom(dump);
363         } catch (InvalidProtocolBufferNanoException ex) {
364             throw new RuntimeException("Failed to parse get-config:\n"
365                     + new String(dump, StandardCharsets.UTF_8), ex);
366         }
367 
368         Configuration config = mContext.getResources().getConfiguration();
369         DisplayManager dm = mContext.getSystemService(DisplayManager.class);
370         Display display = dm.getDisplay(Display.DEFAULT_DISPLAY);
371         DisplayMetrics metrics = new DisplayMetrics();
372         display.getMetrics(metrics);
373 
374         checkResourceConfig(config, metrics, globalConfig.resources);
375         checkDeviceConfig(metrics, globalConfig.device);
376     }
377 
378     @Test
testDeviceConfigWithSecondaryDisplay()379     public void testDeviceConfigWithSecondaryDisplay() throws Exception {
380         VirtualDisplayHelper vd = new VirtualDisplayHelper();
381         final int displayId = vd.setPublicDisplay(true).createAndWaitForDisplay();
382 
383         DisplayManager dm = mContext.getSystemService(DisplayManager.class);
384         Display display = dm.getDisplay(displayId);
385         DisplayMetrics metrics = new DisplayMetrics();
386         display.getMetrics(metrics);
387         String cmd = "cmd activity get-config --proto --device --display " + displayId;
388         byte[] dump = executeShellCommand(cmd);
389 
390         GlobalConfigurationProto globalConfig;
391         globalConfig = GlobalConfigurationProto.parseFrom(dump);
392         Configuration config = mContext.getResources().getConfiguration();
393 
394         checkResourceConfig(config, metrics, globalConfig.resources);
395         checkDeviceConfig(metrics, globalConfig.device);
396 
397         vd.releaseDisplay();
398     }
399 }
400