1 /*
2  * Copyright (C) 2015 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 #include "ConfigDescription.h"
18 
19 #include <string>
20 #include <vector>
21 
22 #include "androidfw/ResourceTypes.h"
23 #include "androidfw/StringPiece.h"
24 
25 #include "Locale.h"
26 #include "SdkConstants.h"
27 #include "util/Util.h"
28 
29 using android::ResTable_config;
30 using android::StringPiece;
31 
32 namespace aapt {
33 
34 static const char* kWildcardName = "any";
35 
DefaultConfig()36 const ConfigDescription& ConfigDescription::DefaultConfig() {
37   static ConfigDescription config = {};
38   return config;
39 }
40 
parseMcc(const char * name,ResTable_config * out)41 static bool parseMcc(const char* name, ResTable_config* out) {
42   if (strcmp(name, kWildcardName) == 0) {
43     if (out) out->mcc = 0;
44     return true;
45   }
46   const char* c = name;
47   if (tolower(*c) != 'm') return false;
48   c++;
49   if (tolower(*c) != 'c') return false;
50   c++;
51   if (tolower(*c) != 'c') return false;
52   c++;
53 
54   const char* val = c;
55 
56   while (*c >= '0' && *c <= '9') {
57     c++;
58   }
59   if (*c != 0) return false;
60   if (c - val != 3) return false;
61 
62   int d = atoi(val);
63   if (d != 0) {
64     if (out) out->mcc = d;
65     return true;
66   }
67 
68   return false;
69 }
70 
parseMnc(const char * name,ResTable_config * out)71 static bool parseMnc(const char* name, ResTable_config* out) {
72   if (strcmp(name, kWildcardName) == 0) {
73     if (out) out->mcc = 0;
74     return true;
75   }
76   const char* c = name;
77   if (tolower(*c) != 'm') return false;
78   c++;
79   if (tolower(*c) != 'n') return false;
80   c++;
81   if (tolower(*c) != 'c') return false;
82   c++;
83 
84   const char* val = c;
85 
86   while (*c >= '0' && *c <= '9') {
87     c++;
88   }
89   if (*c != 0) return false;
90   if (c - val == 0 || c - val > 3) return false;
91 
92   if (out) {
93     out->mnc = atoi(val);
94     if (out->mnc == 0) {
95       out->mnc = ACONFIGURATION_MNC_ZERO;
96     }
97   }
98 
99   return true;
100 }
101 
parseLayoutDirection(const char * name,ResTable_config * out)102 static bool parseLayoutDirection(const char* name, ResTable_config* out) {
103   if (strcmp(name, kWildcardName) == 0) {
104     if (out)
105       out->screenLayout =
106           (out->screenLayout & ~ResTable_config::MASK_LAYOUTDIR) |
107           ResTable_config::LAYOUTDIR_ANY;
108     return true;
109   } else if (strcmp(name, "ldltr") == 0) {
110     if (out)
111       out->screenLayout =
112           (out->screenLayout & ~ResTable_config::MASK_LAYOUTDIR) |
113           ResTable_config::LAYOUTDIR_LTR;
114     return true;
115   } else if (strcmp(name, "ldrtl") == 0) {
116     if (out)
117       out->screenLayout =
118           (out->screenLayout & ~ResTable_config::MASK_LAYOUTDIR) |
119           ResTable_config::LAYOUTDIR_RTL;
120     return true;
121   }
122 
123   return false;
124 }
125 
parseScreenLayoutSize(const char * name,ResTable_config * out)126 static bool parseScreenLayoutSize(const char* name, ResTable_config* out) {
127   if (strcmp(name, kWildcardName) == 0) {
128     if (out)
129       out->screenLayout =
130           (out->screenLayout & ~ResTable_config::MASK_SCREENSIZE) |
131           ResTable_config::SCREENSIZE_ANY;
132     return true;
133   } else if (strcmp(name, "small") == 0) {
134     if (out)
135       out->screenLayout =
136           (out->screenLayout & ~ResTable_config::MASK_SCREENSIZE) |
137           ResTable_config::SCREENSIZE_SMALL;
138     return true;
139   } else if (strcmp(name, "normal") == 0) {
140     if (out)
141       out->screenLayout =
142           (out->screenLayout & ~ResTable_config::MASK_SCREENSIZE) |
143           ResTable_config::SCREENSIZE_NORMAL;
144     return true;
145   } else if (strcmp(name, "large") == 0) {
146     if (out)
147       out->screenLayout =
148           (out->screenLayout & ~ResTable_config::MASK_SCREENSIZE) |
149           ResTable_config::SCREENSIZE_LARGE;
150     return true;
151   } else if (strcmp(name, "xlarge") == 0) {
152     if (out)
153       out->screenLayout =
154           (out->screenLayout & ~ResTable_config::MASK_SCREENSIZE) |
155           ResTable_config::SCREENSIZE_XLARGE;
156     return true;
157   }
158 
159   return false;
160 }
161 
parseScreenLayoutLong(const char * name,ResTable_config * out)162 static bool parseScreenLayoutLong(const char* name, ResTable_config* out) {
163   if (strcmp(name, kWildcardName) == 0) {
164     if (out)
165       out->screenLayout =
166           (out->screenLayout & ~ResTable_config::MASK_SCREENLONG) |
167           ResTable_config::SCREENLONG_ANY;
168     return true;
169   } else if (strcmp(name, "long") == 0) {
170     if (out)
171       out->screenLayout =
172           (out->screenLayout & ~ResTable_config::MASK_SCREENLONG) |
173           ResTable_config::SCREENLONG_YES;
174     return true;
175   } else if (strcmp(name, "notlong") == 0) {
176     if (out)
177       out->screenLayout =
178           (out->screenLayout & ~ResTable_config::MASK_SCREENLONG) |
179           ResTable_config::SCREENLONG_NO;
180     return true;
181   }
182 
183   return false;
184 }
185 
parseScreenRound(const char * name,ResTable_config * out)186 static bool parseScreenRound(const char* name, ResTable_config* out) {
187   if (strcmp(name, kWildcardName) == 0) {
188     if (out)
189       out->screenLayout2 =
190           (out->screenLayout2 & ~ResTable_config::MASK_SCREENROUND) |
191           ResTable_config::SCREENROUND_ANY;
192     return true;
193   } else if (strcmp(name, "round") == 0) {
194     if (out)
195       out->screenLayout2 =
196           (out->screenLayout2 & ~ResTable_config::MASK_SCREENROUND) |
197           ResTable_config::SCREENROUND_YES;
198     return true;
199   } else if (strcmp(name, "notround") == 0) {
200     if (out)
201       out->screenLayout2 =
202           (out->screenLayout2 & ~ResTable_config::MASK_SCREENROUND) |
203           ResTable_config::SCREENROUND_NO;
204     return true;
205   }
206   return false;
207 }
208 
parseWideColorGamut(const char * name,ResTable_config * out)209 static bool parseWideColorGamut(const char* name, ResTable_config* out) {
210   if (strcmp(name, kWildcardName) == 0) {
211     if (out)
212       out->colorMode =
213           (out->colorMode & ~ResTable_config::MASK_WIDE_COLOR_GAMUT) |
214           ResTable_config::WIDE_COLOR_GAMUT_ANY;
215     return true;
216   } else if (strcmp(name, "widecg") == 0) {
217     if (out)
218       out->colorMode =
219           (out->colorMode & ~ResTable_config::MASK_WIDE_COLOR_GAMUT) |
220           ResTable_config::WIDE_COLOR_GAMUT_YES;
221     return true;
222   } else if (strcmp(name, "nowidecg") == 0) {
223     if (out)
224       out->colorMode =
225           (out->colorMode & ~ResTable_config::MASK_WIDE_COLOR_GAMUT) |
226           ResTable_config::WIDE_COLOR_GAMUT_NO;
227     return true;
228   }
229   return false;
230 }
231 
parseHdr(const char * name,ResTable_config * out)232 static bool parseHdr(const char* name, ResTable_config* out) {
233   if (strcmp(name, kWildcardName) == 0) {
234     if (out)
235       out->colorMode =
236           (out->colorMode & ~ResTable_config::MASK_HDR) |
237           ResTable_config::HDR_ANY;
238     return true;
239   } else if (strcmp(name, "highdr") == 0) {
240     if (out)
241       out->colorMode =
242           (out->colorMode & ~ResTable_config::MASK_HDR) |
243           ResTable_config::HDR_YES;
244     return true;
245   } else if (strcmp(name, "lowdr") == 0) {
246     if (out)
247       out->colorMode =
248           (out->colorMode & ~ResTable_config::MASK_HDR) |
249           ResTable_config::HDR_NO;
250     return true;
251   }
252   return false;
253 }
254 
parseOrientation(const char * name,ResTable_config * out)255 static bool parseOrientation(const char* name, ResTable_config* out) {
256   if (strcmp(name, kWildcardName) == 0) {
257     if (out) out->orientation = out->ORIENTATION_ANY;
258     return true;
259   } else if (strcmp(name, "port") == 0) {
260     if (out) out->orientation = out->ORIENTATION_PORT;
261     return true;
262   } else if (strcmp(name, "land") == 0) {
263     if (out) out->orientation = out->ORIENTATION_LAND;
264     return true;
265   } else if (strcmp(name, "square") == 0) {
266     if (out) out->orientation = out->ORIENTATION_SQUARE;
267     return true;
268   }
269 
270   return false;
271 }
272 
parseUiModeType(const char * name,ResTable_config * out)273 static bool parseUiModeType(const char* name, ResTable_config* out) {
274   if (strcmp(name, kWildcardName) == 0) {
275     if (out)
276       out->uiMode = (out->uiMode & ~ResTable_config::MASK_UI_MODE_TYPE) |
277                     ResTable_config::UI_MODE_TYPE_ANY;
278     return true;
279   } else if (strcmp(name, "desk") == 0) {
280     if (out)
281       out->uiMode = (out->uiMode & ~ResTable_config::MASK_UI_MODE_TYPE) |
282                     ResTable_config::UI_MODE_TYPE_DESK;
283     return true;
284   } else if (strcmp(name, "car") == 0) {
285     if (out)
286       out->uiMode = (out->uiMode & ~ResTable_config::MASK_UI_MODE_TYPE) |
287                     ResTable_config::UI_MODE_TYPE_CAR;
288     return true;
289   } else if (strcmp(name, "television") == 0) {
290     if (out)
291       out->uiMode = (out->uiMode & ~ResTable_config::MASK_UI_MODE_TYPE) |
292                     ResTable_config::UI_MODE_TYPE_TELEVISION;
293     return true;
294   } else if (strcmp(name, "appliance") == 0) {
295     if (out)
296       out->uiMode = (out->uiMode & ~ResTable_config::MASK_UI_MODE_TYPE) |
297                     ResTable_config::UI_MODE_TYPE_APPLIANCE;
298     return true;
299   } else if (strcmp(name, "watch") == 0) {
300     if (out)
301       out->uiMode = (out->uiMode & ~ResTable_config::MASK_UI_MODE_TYPE) |
302                     ResTable_config::UI_MODE_TYPE_WATCH;
303     return true;
304   } else if (strcmp(name, "vrheadset") == 0) {
305     if (out)
306       out->uiMode = (out->uiMode & ~ResTable_config::MASK_UI_MODE_TYPE) |
307                     ResTable_config::UI_MODE_TYPE_VR_HEADSET;
308     return true;
309   }
310 
311   return false;
312 }
313 
parseUiModeNight(const char * name,ResTable_config * out)314 static bool parseUiModeNight(const char* name, ResTable_config* out) {
315   if (strcmp(name, kWildcardName) == 0) {
316     if (out)
317       out->uiMode = (out->uiMode & ~ResTable_config::MASK_UI_MODE_NIGHT) |
318                     ResTable_config::UI_MODE_NIGHT_ANY;
319     return true;
320   } else if (strcmp(name, "night") == 0) {
321     if (out)
322       out->uiMode = (out->uiMode & ~ResTable_config::MASK_UI_MODE_NIGHT) |
323                     ResTable_config::UI_MODE_NIGHT_YES;
324     return true;
325   } else if (strcmp(name, "notnight") == 0) {
326     if (out)
327       out->uiMode = (out->uiMode & ~ResTable_config::MASK_UI_MODE_NIGHT) |
328                     ResTable_config::UI_MODE_NIGHT_NO;
329     return true;
330   }
331 
332   return false;
333 }
334 
parseDensity(const char * name,ResTable_config * out)335 static bool parseDensity(const char* name, ResTable_config* out) {
336   if (strcmp(name, kWildcardName) == 0) {
337     if (out) out->density = ResTable_config::DENSITY_DEFAULT;
338     return true;
339   }
340 
341   if (strcmp(name, "anydpi") == 0) {
342     if (out) out->density = ResTable_config::DENSITY_ANY;
343     return true;
344   }
345 
346   if (strcmp(name, "nodpi") == 0) {
347     if (out) out->density = ResTable_config::DENSITY_NONE;
348     return true;
349   }
350 
351   if (strcmp(name, "ldpi") == 0) {
352     if (out) out->density = ResTable_config::DENSITY_LOW;
353     return true;
354   }
355 
356   if (strcmp(name, "mdpi") == 0) {
357     if (out) out->density = ResTable_config::DENSITY_MEDIUM;
358     return true;
359   }
360 
361   if (strcmp(name, "tvdpi") == 0) {
362     if (out) out->density = ResTable_config::DENSITY_TV;
363     return true;
364   }
365 
366   if (strcmp(name, "hdpi") == 0) {
367     if (out) out->density = ResTable_config::DENSITY_HIGH;
368     return true;
369   }
370 
371   if (strcmp(name, "xhdpi") == 0) {
372     if (out) out->density = ResTable_config::DENSITY_XHIGH;
373     return true;
374   }
375 
376   if (strcmp(name, "xxhdpi") == 0) {
377     if (out) out->density = ResTable_config::DENSITY_XXHIGH;
378     return true;
379   }
380 
381   if (strcmp(name, "xxxhdpi") == 0) {
382     if (out) out->density = ResTable_config::DENSITY_XXXHIGH;
383     return true;
384   }
385 
386   char* c = (char*)name;
387   while (*c >= '0' && *c <= '9') {
388     c++;
389   }
390 
391   // check that we have 'dpi' after the last digit.
392   if (toupper(c[0]) != 'D' || toupper(c[1]) != 'P' || toupper(c[2]) != 'I' ||
393       c[3] != 0) {
394     return false;
395   }
396 
397   // temporarily replace the first letter with \0 to
398   // use atoi.
399   char tmp = c[0];
400   c[0] = '\0';
401 
402   int d = atoi(name);
403   c[0] = tmp;
404 
405   if (d != 0) {
406     if (out) out->density = d;
407     return true;
408   }
409 
410   return false;
411 }
412 
parseTouchscreen(const char * name,ResTable_config * out)413 static bool parseTouchscreen(const char* name, ResTable_config* out) {
414   if (strcmp(name, kWildcardName) == 0) {
415     if (out) out->touchscreen = out->TOUCHSCREEN_ANY;
416     return true;
417   } else if (strcmp(name, "notouch") == 0) {
418     if (out) out->touchscreen = out->TOUCHSCREEN_NOTOUCH;
419     return true;
420   } else if (strcmp(name, "stylus") == 0) {
421     if (out) out->touchscreen = out->TOUCHSCREEN_STYLUS;
422     return true;
423   } else if (strcmp(name, "finger") == 0) {
424     if (out) out->touchscreen = out->TOUCHSCREEN_FINGER;
425     return true;
426   }
427 
428   return false;
429 }
430 
parseKeysHidden(const char * name,ResTable_config * out)431 static bool parseKeysHidden(const char* name, ResTable_config* out) {
432   uint8_t mask = 0;
433   uint8_t value = 0;
434   if (strcmp(name, kWildcardName) == 0) {
435     mask = ResTable_config::MASK_KEYSHIDDEN;
436     value = ResTable_config::KEYSHIDDEN_ANY;
437   } else if (strcmp(name, "keysexposed") == 0) {
438     mask = ResTable_config::MASK_KEYSHIDDEN;
439     value = ResTable_config::KEYSHIDDEN_NO;
440   } else if (strcmp(name, "keyshidden") == 0) {
441     mask = ResTable_config::MASK_KEYSHIDDEN;
442     value = ResTable_config::KEYSHIDDEN_YES;
443   } else if (strcmp(name, "keyssoft") == 0) {
444     mask = ResTable_config::MASK_KEYSHIDDEN;
445     value = ResTable_config::KEYSHIDDEN_SOFT;
446   }
447 
448   if (mask != 0) {
449     if (out) out->inputFlags = (out->inputFlags & ~mask) | value;
450     return true;
451   }
452 
453   return false;
454 }
455 
parseKeyboard(const char * name,ResTable_config * out)456 static bool parseKeyboard(const char* name, ResTable_config* out) {
457   if (strcmp(name, kWildcardName) == 0) {
458     if (out) out->keyboard = out->KEYBOARD_ANY;
459     return true;
460   } else if (strcmp(name, "nokeys") == 0) {
461     if (out) out->keyboard = out->KEYBOARD_NOKEYS;
462     return true;
463   } else if (strcmp(name, "qwerty") == 0) {
464     if (out) out->keyboard = out->KEYBOARD_QWERTY;
465     return true;
466   } else if (strcmp(name, "12key") == 0) {
467     if (out) out->keyboard = out->KEYBOARD_12KEY;
468     return true;
469   }
470 
471   return false;
472 }
473 
parseNavHidden(const char * name,ResTable_config * out)474 static bool parseNavHidden(const char* name, ResTable_config* out) {
475   uint8_t mask = 0;
476   uint8_t value = 0;
477   if (strcmp(name, kWildcardName) == 0) {
478     mask = ResTable_config::MASK_NAVHIDDEN;
479     value = ResTable_config::NAVHIDDEN_ANY;
480   } else if (strcmp(name, "navexposed") == 0) {
481     mask = ResTable_config::MASK_NAVHIDDEN;
482     value = ResTable_config::NAVHIDDEN_NO;
483   } else if (strcmp(name, "navhidden") == 0) {
484     mask = ResTable_config::MASK_NAVHIDDEN;
485     value = ResTable_config::NAVHIDDEN_YES;
486   }
487 
488   if (mask != 0) {
489     if (out) out->inputFlags = (out->inputFlags & ~mask) | value;
490     return true;
491   }
492 
493   return false;
494 }
495 
parseNavigation(const char * name,ResTable_config * out)496 static bool parseNavigation(const char* name, ResTable_config* out) {
497   if (strcmp(name, kWildcardName) == 0) {
498     if (out) out->navigation = out->NAVIGATION_ANY;
499     return true;
500   } else if (strcmp(name, "nonav") == 0) {
501     if (out) out->navigation = out->NAVIGATION_NONAV;
502     return true;
503   } else if (strcmp(name, "dpad") == 0) {
504     if (out) out->navigation = out->NAVIGATION_DPAD;
505     return true;
506   } else if (strcmp(name, "trackball") == 0) {
507     if (out) out->navigation = out->NAVIGATION_TRACKBALL;
508     return true;
509   } else if (strcmp(name, "wheel") == 0) {
510     if (out) out->navigation = out->NAVIGATION_WHEEL;
511     return true;
512   }
513 
514   return false;
515 }
516 
parseScreenSize(const char * name,ResTable_config * out)517 static bool parseScreenSize(const char* name, ResTable_config* out) {
518   if (strcmp(name, kWildcardName) == 0) {
519     if (out) {
520       out->screenWidth = out->SCREENWIDTH_ANY;
521       out->screenHeight = out->SCREENHEIGHT_ANY;
522     }
523     return true;
524   }
525 
526   const char* x = name;
527   while (*x >= '0' && *x <= '9') x++;
528   if (x == name || *x != 'x') return false;
529   std::string xName(name, x - name);
530   x++;
531 
532   const char* y = x;
533   while (*y >= '0' && *y <= '9') y++;
534   if (y == name || *y != 0) return false;
535   std::string yName(x, y - x);
536 
537   uint16_t w = (uint16_t)atoi(xName.c_str());
538   uint16_t h = (uint16_t)atoi(yName.c_str());
539   if (w < h) {
540     return false;
541   }
542 
543   if (out) {
544     out->screenWidth = w;
545     out->screenHeight = h;
546   }
547 
548   return true;
549 }
550 
parseSmallestScreenWidthDp(const char * name,ResTable_config * out)551 static bool parseSmallestScreenWidthDp(const char* name, ResTable_config* out) {
552   if (strcmp(name, kWildcardName) == 0) {
553     if (out) {
554       out->smallestScreenWidthDp = out->SCREENWIDTH_ANY;
555     }
556     return true;
557   }
558 
559   if (*name != 's') return false;
560   name++;
561   if (*name != 'w') return false;
562   name++;
563   const char* x = name;
564   while (*x >= '0' && *x <= '9') x++;
565   if (x == name || x[0] != 'd' || x[1] != 'p' || x[2] != 0) return false;
566   std::string xName(name, x - name);
567 
568   if (out) {
569     out->smallestScreenWidthDp = (uint16_t)atoi(xName.c_str());
570   }
571 
572   return true;
573 }
574 
parseScreenWidthDp(const char * name,ResTable_config * out)575 static bool parseScreenWidthDp(const char* name, ResTable_config* out) {
576   if (strcmp(name, kWildcardName) == 0) {
577     if (out) {
578       out->screenWidthDp = out->SCREENWIDTH_ANY;
579     }
580     return true;
581   }
582 
583   if (*name != 'w') return false;
584   name++;
585   const char* x = name;
586   while (*x >= '0' && *x <= '9') x++;
587   if (x == name || x[0] != 'd' || x[1] != 'p' || x[2] != 0) return false;
588   std::string xName(name, x - name);
589 
590   if (out) {
591     out->screenWidthDp = (uint16_t)atoi(xName.c_str());
592   }
593 
594   return true;
595 }
596 
parseScreenHeightDp(const char * name,ResTable_config * out)597 static bool parseScreenHeightDp(const char* name, ResTable_config* out) {
598   if (strcmp(name, kWildcardName) == 0) {
599     if (out) {
600       out->screenHeightDp = out->SCREENWIDTH_ANY;
601     }
602     return true;
603   }
604 
605   if (*name != 'h') return false;
606   name++;
607   const char* x = name;
608   while (*x >= '0' && *x <= '9') x++;
609   if (x == name || x[0] != 'd' || x[1] != 'p' || x[2] != 0) return false;
610   std::string xName(name, x - name);
611 
612   if (out) {
613     out->screenHeightDp = (uint16_t)atoi(xName.c_str());
614   }
615 
616   return true;
617 }
618 
parseVersion(const char * name,ResTable_config * out)619 static bool parseVersion(const char* name, ResTable_config* out) {
620   if (strcmp(name, kWildcardName) == 0) {
621     if (out) {
622       out->sdkVersion = out->SDKVERSION_ANY;
623       out->minorVersion = out->MINORVERSION_ANY;
624     }
625     return true;
626   }
627 
628   if (*name != 'v') {
629     return false;
630   }
631 
632   name++;
633   const char* s = name;
634   while (*s >= '0' && *s <= '9') s++;
635   if (s == name || *s != 0) return false;
636   std::string sdkName(name, s - name);
637 
638   if (out) {
639     out->sdkVersion = (uint16_t)atoi(sdkName.c_str());
640     out->minorVersion = 0;
641   }
642 
643   return true;
644 }
645 
Parse(const StringPiece & str,ConfigDescription * out)646 bool ConfigDescription::Parse(const StringPiece& str, ConfigDescription* out) {
647   std::vector<std::string> parts = util::SplitAndLowercase(str, '-');
648 
649   ConfigDescription config;
650   ssize_t parts_consumed = 0;
651   LocaleValue locale;
652 
653   const auto parts_end = parts.end();
654   auto part_iter = parts.begin();
655 
656   if (str.size() == 0) {
657     goto success;
658   }
659 
660   if (parseMcc(part_iter->c_str(), &config)) {
661     ++part_iter;
662     if (part_iter == parts_end) {
663       goto success;
664     }
665   }
666 
667   if (parseMnc(part_iter->c_str(), &config)) {
668     ++part_iter;
669     if (part_iter == parts_end) {
670       goto success;
671     }
672   }
673 
674   // Locale spans a few '-' separators, so we let it
675   // control the index.
676   parts_consumed = locale.InitFromParts(part_iter, parts_end);
677   if (parts_consumed < 0) {
678     return false;
679   } else {
680     locale.WriteTo(&config);
681     part_iter += parts_consumed;
682     if (part_iter == parts_end) {
683       goto success;
684     }
685   }
686 
687   if (parseLayoutDirection(part_iter->c_str(), &config)) {
688     ++part_iter;
689     if (part_iter == parts_end) {
690       goto success;
691     }
692   }
693 
694   if (parseSmallestScreenWidthDp(part_iter->c_str(), &config)) {
695     ++part_iter;
696     if (part_iter == parts_end) {
697       goto success;
698     }
699   }
700 
701   if (parseScreenWidthDp(part_iter->c_str(), &config)) {
702     ++part_iter;
703     if (part_iter == parts_end) {
704       goto success;
705     }
706   }
707 
708   if (parseScreenHeightDp(part_iter->c_str(), &config)) {
709     ++part_iter;
710     if (part_iter == parts_end) {
711       goto success;
712     }
713   }
714 
715   if (parseScreenLayoutSize(part_iter->c_str(), &config)) {
716     ++part_iter;
717     if (part_iter == parts_end) {
718       goto success;
719     }
720   }
721 
722   if (parseScreenLayoutLong(part_iter->c_str(), &config)) {
723     ++part_iter;
724     if (part_iter == parts_end) {
725       goto success;
726     }
727   }
728 
729   if (parseScreenRound(part_iter->c_str(), &config)) {
730     ++part_iter;
731     if (part_iter == parts_end) {
732       goto success;
733     }
734   }
735 
736   if (parseWideColorGamut(part_iter->c_str(), &config)) {
737     ++part_iter;
738     if (part_iter == parts_end) {
739       goto success;
740     }
741   }
742 
743   if (parseHdr(part_iter->c_str(), &config)) {
744     ++part_iter;
745     if (part_iter == parts_end) {
746       goto success;
747     }
748   }
749 
750   if (parseOrientation(part_iter->c_str(), &config)) {
751     ++part_iter;
752     if (part_iter == parts_end) {
753       goto success;
754     }
755   }
756 
757   if (parseUiModeType(part_iter->c_str(), &config)) {
758     ++part_iter;
759     if (part_iter == parts_end) {
760       goto success;
761     }
762   }
763 
764   if (parseUiModeNight(part_iter->c_str(), &config)) {
765     ++part_iter;
766     if (part_iter == parts_end) {
767       goto success;
768     }
769   }
770 
771   if (parseDensity(part_iter->c_str(), &config)) {
772     ++part_iter;
773     if (part_iter == parts_end) {
774       goto success;
775     }
776   }
777 
778   if (parseTouchscreen(part_iter->c_str(), &config)) {
779     ++part_iter;
780     if (part_iter == parts_end) {
781       goto success;
782     }
783   }
784 
785   if (parseKeysHidden(part_iter->c_str(), &config)) {
786     ++part_iter;
787     if (part_iter == parts_end) {
788       goto success;
789     }
790   }
791 
792   if (parseKeyboard(part_iter->c_str(), &config)) {
793     ++part_iter;
794     if (part_iter == parts_end) {
795       goto success;
796     }
797   }
798 
799   if (parseNavHidden(part_iter->c_str(), &config)) {
800     ++part_iter;
801     if (part_iter == parts_end) {
802       goto success;
803     }
804   }
805 
806   if (parseNavigation(part_iter->c_str(), &config)) {
807     ++part_iter;
808     if (part_iter == parts_end) {
809       goto success;
810     }
811   }
812 
813   if (parseScreenSize(part_iter->c_str(), &config)) {
814     ++part_iter;
815     if (part_iter == parts_end) {
816       goto success;
817     }
818   }
819 
820   if (parseVersion(part_iter->c_str(), &config)) {
821     ++part_iter;
822     if (part_iter == parts_end) {
823       goto success;
824     }
825   }
826 
827   // Unrecognized.
828   return false;
829 
830 success:
831   if (out != NULL) {
832     ApplyVersionForCompatibility(&config);
833     *out = config;
834   }
835   return true;
836 }
837 
ApplyVersionForCompatibility(ConfigDescription * config)838 void ConfigDescription::ApplyVersionForCompatibility(
839     ConfigDescription* config) {
840   uint16_t min_sdk = 0;
841   if ((config->uiMode & ResTable_config::MASK_UI_MODE_TYPE)
842                 == ResTable_config::UI_MODE_TYPE_VR_HEADSET ||
843             config->colorMode & ResTable_config::MASK_WIDE_COLOR_GAMUT ||
844             config->colorMode & ResTable_config::MASK_HDR) {
845         min_sdk = SDK_O;
846   } else if (config->screenLayout2 & ResTable_config::MASK_SCREENROUND) {
847     min_sdk = SDK_MARSHMALLOW;
848   } else if (config->density == ResTable_config::DENSITY_ANY) {
849     min_sdk = SDK_LOLLIPOP;
850   } else if (config->smallestScreenWidthDp !=
851                  ResTable_config::SCREENWIDTH_ANY ||
852              config->screenWidthDp != ResTable_config::SCREENWIDTH_ANY ||
853              config->screenHeightDp != ResTable_config::SCREENHEIGHT_ANY) {
854     min_sdk = SDK_HONEYCOMB_MR2;
855   } else if ((config->uiMode & ResTable_config::MASK_UI_MODE_TYPE) !=
856                  ResTable_config::UI_MODE_TYPE_ANY ||
857              (config->uiMode & ResTable_config::MASK_UI_MODE_NIGHT) !=
858                  ResTable_config::UI_MODE_NIGHT_ANY) {
859     min_sdk = SDK_FROYO;
860   } else if ((config->screenLayout & ResTable_config::MASK_SCREENSIZE) !=
861                  ResTable_config::SCREENSIZE_ANY ||
862              (config->screenLayout & ResTable_config::MASK_SCREENLONG) !=
863                  ResTable_config::SCREENLONG_ANY ||
864              config->density != ResTable_config::DENSITY_DEFAULT) {
865     min_sdk = SDK_DONUT;
866   }
867 
868   if (min_sdk > config->sdkVersion) {
869     config->sdkVersion = min_sdk;
870   }
871 }
872 
CopyWithoutSdkVersion() const873 ConfigDescription ConfigDescription::CopyWithoutSdkVersion() const {
874   ConfigDescription copy = *this;
875   copy.sdkVersion = 0;
876   return copy;
877 }
878 
Dominates(const ConfigDescription & o) const879 bool ConfigDescription::Dominates(const ConfigDescription& o) const {
880   if (*this == DefaultConfig() || *this == o) {
881     return true;
882   }
883   return MatchWithDensity(o) && !o.MatchWithDensity(*this) &&
884          !isMoreSpecificThan(o) && !o.HasHigherPrecedenceThan(*this);
885 }
886 
HasHigherPrecedenceThan(const ConfigDescription & o) const887 bool ConfigDescription::HasHigherPrecedenceThan(
888     const ConfigDescription& o) const {
889   // The order of the following tests defines the importance of one
890   // configuration parameter over another. Those tests first are more
891   // important, trumping any values in those following them.
892   // The ordering should be the same as ResTable_config#isBetterThan.
893   if (mcc || o.mcc) return (!o.mcc);
894   if (mnc || o.mnc) return (!o.mnc);
895   if (language[0] || o.language[0]) return (!o.language[0]);
896   if (country[0] || o.country[0]) return (!o.country[0]);
897   // Script and variant require either a language or country, both of which
898   // have higher precedence.
899   if ((screenLayout | o.screenLayout) & MASK_LAYOUTDIR) {
900     return !(o.screenLayout & MASK_LAYOUTDIR);
901   }
902   if (smallestScreenWidthDp || o.smallestScreenWidthDp)
903     return (!o.smallestScreenWidthDp);
904   if (screenWidthDp || o.screenWidthDp) return (!o.screenWidthDp);
905   if (screenHeightDp || o.screenHeightDp) return (!o.screenHeightDp);
906   if ((screenLayout | o.screenLayout) & MASK_SCREENSIZE) {
907     return !(o.screenLayout & MASK_SCREENSIZE);
908   }
909   if ((screenLayout | o.screenLayout) & MASK_SCREENLONG) {
910     return !(o.screenLayout & MASK_SCREENLONG);
911   }
912   if ((screenLayout2 | o.screenLayout2) & MASK_SCREENROUND) {
913     return !(o.screenLayout2 & MASK_SCREENROUND);
914   }
915   if ((colorMode | o.colorMode) & MASK_HDR) {
916     return !(o.colorMode & MASK_HDR);
917   }
918   if ((colorMode | o.colorMode) & MASK_WIDE_COLOR_GAMUT) {
919     return !(o.colorMode & MASK_WIDE_COLOR_GAMUT);
920   }
921   if (orientation || o.orientation) return (!o.orientation);
922   if ((uiMode | o.uiMode) & MASK_UI_MODE_TYPE) {
923     return !(o.uiMode & MASK_UI_MODE_TYPE);
924   }
925   if ((uiMode | o.uiMode) & MASK_UI_MODE_NIGHT) {
926     return !(o.uiMode & MASK_UI_MODE_NIGHT);
927   }
928   if (density || o.density) return (!o.density);
929   if (touchscreen || o.touchscreen) return (!o.touchscreen);
930   if ((inputFlags | o.inputFlags) & MASK_KEYSHIDDEN) {
931     return !(o.inputFlags & MASK_KEYSHIDDEN);
932   }
933   if ((inputFlags | o.inputFlags) & MASK_NAVHIDDEN) {
934     return !(o.inputFlags & MASK_NAVHIDDEN);
935   }
936   if (keyboard || o.keyboard) return (!o.keyboard);
937   if (navigation || o.navigation) return (!o.navigation);
938   if (screenWidth || o.screenWidth) return (!o.screenWidth);
939   if (screenHeight || o.screenHeight) return (!o.screenHeight);
940   if (sdkVersion || o.sdkVersion) return (!o.sdkVersion);
941   if (minorVersion || o.minorVersion) return (!o.minorVersion);
942   // Both configurations have nothing defined except some possible future
943   // value. Returning the comparison of the two configurations is a
944   // "best effort" at this point to protect against incorrect dominations.
945   return *this != o;
946 }
947 
ConflictsWith(const ConfigDescription & o) const948 bool ConfigDescription::ConflictsWith(const ConfigDescription& o) const {
949   // This method should be updated as new configuration parameters are
950   // introduced (e.g. screenConfig2).
951   auto pred = [](const uint32_t a, const uint32_t b) -> bool {
952     return a == 0 || b == 0 || a == b;
953   };
954   // The values here can be found in ResTable_config#match. Density and range
955   // values can't lead to conflicts, and are ignored.
956   return !pred(mcc, o.mcc) || !pred(mnc, o.mnc) || !pred(locale, o.locale) ||
957          !pred(screenLayout & MASK_LAYOUTDIR,
958                o.screenLayout & MASK_LAYOUTDIR) ||
959          !pred(screenLayout & MASK_SCREENLONG,
960                o.screenLayout & MASK_SCREENLONG) ||
961          !pred(screenLayout & MASK_UI_MODE_TYPE,
962                o.screenLayout & MASK_UI_MODE_TYPE) ||
963          !pred(uiMode & MASK_UI_MODE_TYPE, o.uiMode & MASK_UI_MODE_TYPE) ||
964          !pred(uiMode & MASK_UI_MODE_NIGHT, o.uiMode & MASK_UI_MODE_NIGHT) ||
965          !pred(screenLayout2 & MASK_SCREENROUND,
966                o.screenLayout2 & MASK_SCREENROUND) ||
967          !pred(colorMode & MASK_HDR, o.colorMode & MASK_HDR) ||
968          !pred(colorMode & MASK_WIDE_COLOR_GAMUT,
969                o.colorMode & MASK_WIDE_COLOR_GAMUT) ||
970          !pred(orientation, o.orientation) ||
971          !pred(touchscreen, o.touchscreen) ||
972          !pred(inputFlags & MASK_KEYSHIDDEN, o.inputFlags & MASK_KEYSHIDDEN) ||
973          !pred(inputFlags & MASK_NAVHIDDEN, o.inputFlags & MASK_NAVHIDDEN) ||
974          !pred(keyboard, o.keyboard) || !pred(navigation, o.navigation);
975 }
976 
IsCompatibleWith(const ConfigDescription & o) const977 bool ConfigDescription::IsCompatibleWith(const ConfigDescription& o) const {
978   return !ConflictsWith(o) && !Dominates(o) && !o.Dominates(*this);
979 }
980 
981 }  // namespace aapt
982