1 /* 2 * Copyright (C) 2020 Google LLC 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 package com.google.carrier; 17 18 import static com.google.common.collect.Multimaps.flatteningToMultimap; 19 import static com.google.common.collect.Multimaps.toMultimap; 20 import static java.nio.charset.StandardCharsets.UTF_8; 21 import static java.util.Comparator.comparing; 22 23 import com.beust.jcommander.JCommander; 24 import com.beust.jcommander.Parameter; 25 import com.beust.jcommander.Parameters; 26 import com.google.auto.value.AutoValue; 27 import com.google.common.base.Ascii; 28 import com.google.common.base.CharMatcher; 29 import com.google.common.collect.ImmutableList; 30 import com.google.common.collect.ImmutableMap; 31 import com.google.common.collect.ImmutableSet; 32 import com.google.common.collect.Multimap; 33 import com.google.common.collect.MultimapBuilder; 34 import com.google.protobuf.Descriptors; 35 import com.google.protobuf.TextFormat; 36 import com.google.carrier.CarrierConfig; 37 import com.google.carrier.CarrierId; 38 import com.google.carrier.CarrierList; 39 import com.google.carrier.CarrierMap; 40 import com.google.carrier.CarrierSettings; 41 import com.google.carrier.IntArray; 42 import com.google.carrier.MultiCarrierSettings; 43 import com.google.carrier.TextArray; 44 import com.android.providers.telephony.CarrierIdProto.CarrierAttribute; 45 import java.io.BufferedReader; 46 import java.io.BufferedWriter; 47 import java.io.File; 48 import java.io.FileInputStream; 49 import java.io.FileOutputStream; 50 import java.io.IOException; 51 import java.io.InputStream; 52 import java.io.InputStreamReader; 53 import java.io.OutputStream; 54 import java.io.OutputStreamWriter; 55 import java.util.ArrayList; 56 import java.util.HashMap; 57 import java.util.List; 58 import java.util.Map; 59 import java.util.TreeMap; 60 import java.util.regex.Matcher; 61 import java.util.regex.Pattern; 62 import javax.xml.parsers.DocumentBuilder; 63 import javax.xml.parsers.DocumentBuilderFactory; 64 import javax.xml.parsers.ParserConfigurationException; 65 import org.w3c.dom.Document; 66 import org.w3c.dom.Element; 67 import org.w3c.dom.NamedNodeMap; 68 import org.w3c.dom.Node; 69 import org.w3c.dom.NodeList; 70 import org.xml.sax.SAXException; 71 72 /** 73 * This command converts carrier config XML into text protobuf. 74 * 75 * <ul> 76 * <li>input: the assets/ from AOSP CarrierConfig app 77 * <li>input: vendor.xml file(s) which override(s) assets 78 * <li>input: a tier-1 carrier list in text protobuf (in --output_dir) 79 * <li>input: the version number for output files 80 * <li>output: an other_carriers.textpb - a list of other (non-tier-1) carriers 81 * <li>output: an others.textpb containing carrier configs for non tier-1 carriers 82 * <li>output: a .textpb for every single tier-1 carrier 83 * </ul> 84 */ 85 @Parameters(separators = "=") 86 public final class CarrierConfigConverterV2 { 87 @Parameter(names = "--assets", description = "The source AOSP assets/ directory.") 88 private String assetsDirName = "/tmp/carrierconfig/assets"; 89 90 @Parameter( 91 names = "--vendor_xml", 92 description = 93 "The source vendor.xml file(s). If multiple files provided, the order decides config" 94 + " precedence, ie. configs in a file are overwritten by configs in files AFTER it.") 95 private List<String> vendorXmlFiles = ImmutableList.of("/tmp/carrierconfig/vendor.xml"); 96 97 @Parameter( 98 names = "--output_dir", 99 description = "The destination data directory, with tier1_carriers.textpb in it.") 100 private String outputDir = "/tmp/carrierconfig/out"; 101 102 @Parameter(names = "--version", description = "The version number for all output textpb.") 103 private long version = 1L; 104 105 // Resource file path to the AOSP carrier list file 106 private static final String RESOURCE_CARRIER_LIST = 107 "/assets/latest_carrier_id/carrier_list.textpb"; 108 109 // Constants used in parsing XMLs. 110 private static final String XML_SUFFIX = ".xml"; 111 private static final String CARRIER_CONFIG_MCCMNC_XML_PREFIX = "carrier_config_mccmnc_"; 112 private static final String CARRIER_CONFIG_CID_XML_PREFIX = "carrier_config_carrierid_"; 113 private static final String KEY_MCCMNC_PREFIX = "mccmnc_"; 114 private static final String KEY_CID_PREFIX = "cid_"; 115 private static final String TAG_CARRIER_CONFIG = "carrier_config"; 116 117 /** Entry point when invoked from command line. */ 118 public static void main(String[] args) throws IOException { 119 CarrierConfigConverterV2 converter = new CarrierConfigConverterV2(); 120 new JCommander(converter, args); 121 converter.convert(); 122 } 123 124 /** Entry point when invoked from other Java code, eg. the server side conversion tool. */ 125 public static void convert( 126 String vendorXmlFile, String assetsDirName, String outputDir, long version) 127 throws IOException { 128 CarrierConfigConverterV2 converter = new CarrierConfigConverterV2(); 129 converter.vendorXmlFiles = ImmutableList.of(vendorXmlFile); 130 converter.assetsDirName = assetsDirName; 131 converter.outputDir = outputDir; 132 converter.version = version; 133 converter.convert(); 134 } 135 136 private void convert() throws IOException { 137 String carriersTextpbFile = getPathAsString(outputDir, "tier1_carriers.textpb"); 138 String settingsTextpbDir = getPathAsString(outputDir, "setting"); 139 CarrierList tier1Carriers; 140 ArrayList<CarrierMap> otherCarriers = new ArrayList<>(); 141 ArrayList<String> outFiles = new ArrayList<>(); 142 HashMap<CarrierId, Map<String, CarrierConfig.Config>> rawConfigs = new HashMap<>(); 143 TreeMap<String, CarrierConfig> tier1Configs = new TreeMap<>(); 144 TreeMap<String, CarrierConfig> othersConfigs = new TreeMap<>(); 145 DocumentBuilder xmlDocBuilder = getDocumentBuilder(); 146 Multimap<Integer, CarrierId> aospCarrierList = loadAospCarrierList(); 147 Multimap<CarrierId, Integer> reverseAospCarrierList = reverseAospCarrierList(aospCarrierList); 148 149 /* 150 * High-level flow: 151 * 1. Parse all input XMLs into memory 152 * 2. Collect a list of interested carriers from input, represented by CarrierId. 153 * 2. For each CarrierId, build its carreir configs, following AOSP DefaultCarrierConfigService. 154 * 3. Merge CarrierId's as per tier1_carriers.textpb 155 */ 156 157 // 1. Parse all input XMLs into memory 158 Map<String, Document> assetsXmls = new HashMap<>(); 159 List<Document> vendorXmls = new ArrayList<>(); 160 // Parse assets/carrier_config_*.xml 161 for (File childFile : new File(assetsDirName).listFiles()) { 162 String childFileName = childFile.getName(); 163 String fullChildName = childFile.getCanonicalPath(); 164 if (childFileName.startsWith(CARRIER_CONFIG_MCCMNC_XML_PREFIX)) { 165 String mccMnc = 166 childFileName.substring( 167 CARRIER_CONFIG_MCCMNC_XML_PREFIX.length(), childFileName.indexOf(XML_SUFFIX)); 168 if (!mccMnc.matches("\\d{5,6}")) { 169 throw new IOException("Invalid mcc/mnc " + mccMnc + " found in " + childFileName); 170 } 171 try { 172 assetsXmls.put(KEY_MCCMNC_PREFIX + mccMnc, parseXmlDoc(fullChildName, xmlDocBuilder)); 173 } catch (SAXException | IOException e) { 174 throw new IOException("Failed to parse " + childFileName, e); 175 } 176 } else if (childFileName.startsWith(CARRIER_CONFIG_CID_XML_PREFIX)) { 177 String cidAndCarrierName = 178 childFileName.substring( 179 CARRIER_CONFIG_CID_XML_PREFIX.length(), childFileName.indexOf(XML_SUFFIX)); 180 int cid = -1; 181 try { 182 cid = Integer.parseInt(cidAndCarrierName.split("_", -1)[0]); 183 } catch (NumberFormatException e) { 184 throw new IOException("Invalid carrierid found in " + childFileName, e); 185 } 186 try { 187 assetsXmls.put(KEY_CID_PREFIX + cid, parseXmlDoc(fullChildName, xmlDocBuilder)); 188 } catch (SAXException | IOException e) { 189 throw new IOException("Failed to parse " + childFileName, e); 190 } 191 } 192 // ignore other malformatted files. 193 } 194 // Parse vendor.xml files 195 for (String vendorXmlFile : vendorXmlFiles) { 196 try { 197 vendorXmls.add(parseXmlDoc(vendorXmlFile, xmlDocBuilder)); 198 } catch (SAXException | IOException e) { 199 throw new IOException("Failed to parse " + vendorXmlFile, e); 200 } 201 } 202 203 // 2. Collect all carriers from input, represented by CarrierId. 204 List<CarrierId> carriers = new ArrayList<>(); 205 // Traverse <carrier_config /> labels in each file. 206 for (Map.Entry<String, Document> xml : assetsXmls.entrySet()) { 207 if (xml.getKey().startsWith(KEY_MCCMNC_PREFIX)) { 208 String mccMnc = xml.getKey().substring(KEY_MCCMNC_PREFIX.length()); 209 for (Element element : getElementsByTagName(xml.getValue(), TAG_CARRIER_CONFIG)) { 210 try { 211 CarrierId id = parseCarrierId(element).setMccMnc(mccMnc).build(); 212 carriers.add(id); 213 } catch (UnsupportedOperationException e) { 214 throw new IOException("Unsupported syntax in assets/ for " + mccMnc, e); 215 } 216 } 217 } else if (xml.getKey().startsWith(KEY_CID_PREFIX)) { 218 int cid = Integer.parseInt(xml.getKey().substring(KEY_CID_PREFIX.length())); 219 if (aospCarrierList.containsKey(cid)) { 220 carriers.addAll(aospCarrierList.get(cid)); 221 } else { 222 System.err.printf("Undefined cid %d in assets/. Ignore.\n", cid); 223 } 224 } 225 } 226 for (Document vendorXml : vendorXmls) { 227 for (Element element : getElementsByTagName(vendorXml, TAG_CARRIER_CONFIG)) { 228 // First, try to parse cid 229 if (element.hasAttribute("cid")) { 230 String cidAsString = element.getAttribute("cid"); 231 int cid = Integer.parseInt(cidAsString); 232 if (aospCarrierList.containsKey(cid)) { 233 carriers.addAll(aospCarrierList.get(cid)); 234 } else { 235 System.err.printf("Undefined cid %d in vendor.xml. Ignore.\n", cid); 236 } 237 } else { 238 // Then, try to parse CarrierId 239 CarrierId.Builder id = parseCarrierId(element); 240 // A valid mccmnc is 5- or 6-digit. But vendor.xml see special cases below: 241 // <carrier_config> element may have just "mcc" and not "mnc" for 242 // country-wise config. Such a element doesn't make a carrier; but still keep it so 243 // can be used if a mccmnc appears in APNs later. 244 if (id.getMccMnc().length() == 3) { 245 // special case 246 carriers.add(id.build()); 247 } else if (id.getMccMnc().length() == 5 || id.getMccMnc().length() == 6) { 248 // Normal mcc+mnc 249 carriers.add(id.build()); 250 } else { 251 System.err.printf("Invalid mcc/mnc: %s. Ignore.\n", id.getMccMnc()); 252 } 253 } 254 } 255 } 256 257 // 3. For each CarrierId, build its carrier configs, following AOSP DefaultCarrierConfigService. 258 for (CarrierId carrier : carriers) { 259 Map<String, CarrierConfig.Config> config = ImmutableMap.of(); 260 261 CarrierIdentifier id = getCid(carrier, reverseAospCarrierList); 262 if (id.getCarrierId() != -1) { 263 HashMap<String, CarrierConfig.Config> configBySpecificCarrierId = 264 parseCarrierConfigFromXml( 265 assetsXmls.get(KEY_CID_PREFIX + id.getSpecificCarrierId()), id); 266 HashMap<String, CarrierConfig.Config> configByCarrierId = 267 parseCarrierConfigFromXml(assetsXmls.get(KEY_CID_PREFIX + id.getCarrierId()), id); 268 HashMap<String, CarrierConfig.Config> configByMccMncFallBackCarrierId = 269 parseCarrierConfigFromXml(assetsXmls.get(KEY_CID_PREFIX + id.getMccmncCarrierId()), id); 270 // priority: specific carrier id > carrier id > mccmnc fallback carrier id 271 if (!configBySpecificCarrierId.isEmpty()) { 272 config = configBySpecificCarrierId; 273 } else if (!configByCarrierId.isEmpty()) { 274 config = configByCarrierId; 275 } else if (!configByMccMncFallBackCarrierId.isEmpty()) { 276 config = configByMccMncFallBackCarrierId; 277 } 278 } 279 if (config.isEmpty()) { 280 // fallback to use mccmnc.xml when there is no carrier id named configuration found. 281 config = 282 parseCarrierConfigFromXml(assetsXmls.get(KEY_MCCMNC_PREFIX + carrier.getMccMnc()), id); 283 } 284 // Treat vendor.xml files as if they were appended to the carrier configs read from assets. 285 for (Document vendorXml : vendorXmls) { 286 HashMap<String, CarrierConfig.Config> vendorConfig = 287 parseCarrierConfigFromVendorXml(vendorXml, id); 288 config.putAll(vendorConfig); 289 } 290 291 rawConfigs.put(carrier, config); 292 } 293 294 // Read tier1_carriers.textpb 295 try (InputStream carriersTextpb = new FileInputStream(new File(carriersTextpbFile)); 296 BufferedReader br = new BufferedReader(new InputStreamReader(carriersTextpb, UTF_8))) { 297 CarrierList.Builder builder = CarrierList.newBuilder(); 298 TextFormat.getParser().merge(br, builder); 299 tier1Carriers = builder.build(); 300 } 301 302 // Compose tier1Configs and othersConfigs from rawConfigs 303 rawConfigs.forEach( 304 (carrierId, configs) -> { 305 String cname = getCanonicalName(tier1Carriers, carrierId); 306 CarrierConfig.Builder ccb = toCarrierConfigBuilder(configs); 307 if (cname != null) { // tier-1 carrier 308 if (tier1Configs.containsKey(cname)) { 309 tier1Configs.put( 310 cname, CarrierProtoUtils.mergeCarrierConfig(tier1Configs.get(cname), ccb)); 311 } else { 312 tier1Configs.put(cname, ccb.build()); 313 } 314 } else { // other carrier 315 cname = generateCanonicalNameForOthers(carrierId); 316 otherCarriers.add( 317 CarrierMap.newBuilder().addCarrierId(carrierId).setCanonicalName(cname).build()); 318 othersConfigs.put(cname, ccb.build()); 319 } 320 }); 321 322 // output tier1 carrier settings 323 for (int i = 0; i < tier1Carriers.getEntryCount(); i++) { 324 CarrierMap cm = tier1Carriers.getEntry(i); 325 String cname = cm.getCanonicalName(); 326 String fileName = getPathAsString(settingsTextpbDir, cname + ".textpb"); 327 328 outFiles.add(fileName); 329 330 try (OutputStream os = new FileOutputStream(new File(fileName)); 331 BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(os, UTF_8))) { 332 CarrierSettings.Builder cs = CarrierSettings.newBuilder().setCanonicalName(cname); 333 if (tier1Configs.containsKey(cname)) { 334 cs.setConfigs(sortConfig(tier1Configs.get(cname)).toBuilder().build()); 335 } 336 cs.setVersion(version); 337 TextFormat.printUnicode(cs.build(), bw); 338 } 339 } 340 341 // Output other carriers list 342 String otherCarriersFile = getPathAsString(outputDir, "other_carriers.textpb"); 343 outFiles.add(otherCarriersFile); 344 CarrierProtoUtils.sortCarrierMapEntries(otherCarriers); 345 try (OutputStream os = new FileOutputStream(new File(otherCarriersFile)); 346 BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(os, UTF_8))) { 347 CarrierList cl = 348 CarrierList.newBuilder().addAllEntry(otherCarriers).setVersion(version).build(); 349 TextFormat.printUnicode(cl, bw); 350 } 351 352 // Output other carriers settings 353 String othersFileName = getPathAsString(settingsTextpbDir, "others.textpb"); 354 outFiles.add(othersFileName); 355 try (OutputStream os = new FileOutputStream(new File(othersFileName)); 356 BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(os, UTF_8))) { 357 MultiCarrierSettings.Builder mcs = MultiCarrierSettings.newBuilder().setVersion(version); 358 othersConfigs.forEach( 359 (cname, cc) -> { 360 mcs.addSetting( 361 CarrierSettings.newBuilder() 362 .setCanonicalName(cname) 363 .setConfigs(sortConfig(cc).toBuilder().build()) 364 .build()); 365 }); 366 TextFormat.printUnicode(mcs.build(), bw); 367 } 368 369 // Print out the list of all output file names 370 System.out.println("SUCCESS! Files generated:"); 371 for (String fileName : outFiles) { 372 System.out.println(fileName); 373 } 374 } 375 376 private static DocumentBuilder getDocumentBuilder() { 377 try { 378 DocumentBuilderFactory dbFactory = DocumentBuilderFactory.newInstance(); 379 return dbFactory.newDocumentBuilder(); 380 } catch (ParserConfigurationException e) { 381 throw new IllegalStateException(e); 382 } 383 } 384 385 private static Multimap<Integer, CarrierId> loadAospCarrierList() throws IOException { 386 com.android.providers.telephony.CarrierIdProto.CarrierList.Builder aospCarrierList = 387 com.android.providers.telephony.CarrierIdProto.CarrierList.newBuilder(); 388 try (InputStream textpb = 389 CarrierConfigConverterV2.class.getResourceAsStream(RESOURCE_CARRIER_LIST); 390 BufferedReader textpbReader = new BufferedReader(new InputStreamReader(textpb, UTF_8))) { 391 TextFormat.getParser().merge(textpbReader, aospCarrierList); 392 } 393 return aospCarrierList.getCarrierIdList().stream() 394 .collect( 395 flatteningToMultimap( 396 cid -> cid.getCanonicalId(), 397 cid -> carrierAttributeToCarrierId(cid.getCarrierAttributeList()).stream(), 398 MultimapBuilder.linkedHashKeys().arrayListValues()::build)); 399 } 400 401 // Convert `CarrierAttribute`s to `CarrierId`s. 402 // A CarrierAttribute message with fields not supported by CarrierSettings, like preferred_apn, 403 // is ignored. 404 private static ImmutableList<CarrierId> carrierAttributeToCarrierId( 405 List<CarrierAttribute> carrierAttributes) { 406 List<CarrierId> result = new ArrayList<>(); 407 ImmutableSet<Descriptors.FieldDescriptor> supportedFields = 408 ImmutableSet.of( 409 CarrierAttribute.getDescriptor().findFieldByName("mccmnc_tuple"), 410 CarrierAttribute.getDescriptor().findFieldByName("imsi_prefix_xpattern"), 411 CarrierAttribute.getDescriptor().findFieldByName("spn"), 412 CarrierAttribute.getDescriptor().findFieldByName("gid1")); 413 for (CarrierAttribute carrierAttribute : carrierAttributes) { 414 if (!carrierAttribute.getAllFields().keySet().stream().allMatch(supportedFields::contains)) { 415 // This `CarrierAttribute` contains unsupported fields; skip. 416 continue; 417 } 418 for (String mccmnc : carrierAttribute.getMccmncTupleList()) { 419 CarrierId.Builder carrierId = CarrierId.newBuilder().setMccMnc(mccmnc); 420 if (carrierAttribute.getImsiPrefixXpatternCount() > 0) { 421 for (String imsi : carrierAttribute.getImsiPrefixXpatternList()) { 422 result.add(carrierId.setImsi(imsi).build()); 423 } 424 } else if (carrierAttribute.getGid1Count() > 0) { 425 for (String gid1 : carrierAttribute.getGid1List()) { 426 result.add(carrierId.setGid1(gid1).build()); 427 } 428 } else if (carrierAttribute.getSpnCount() > 0) { 429 for (String spn : carrierAttribute.getSpnList()) { 430 // Some SPN has trailng space character \r, messing up textpb. Remove them. 431 // It won't affect CarrierSettings which uses prefix matching for SPN. 432 result.add(carrierId.setSpn(CharMatcher.whitespace().trimTrailingFrom(spn)).build()); 433 } 434 } else { // Ignore other attributes not supported by CarrierSettings 435 result.add(carrierId.build()); 436 } 437 } 438 } 439 // Dedup 440 return ImmutableSet.copyOf(result).asList(); 441 } 442 443 private static Multimap<CarrierId, Integer> reverseAospCarrierList( 444 Multimap<Integer, CarrierId> aospCarrierList) { 445 return aospCarrierList.entries().stream() 446 .collect( 447 toMultimap( 448 entry -> entry.getValue(), 449 entry -> entry.getKey(), 450 MultimapBuilder.linkedHashKeys().arrayListValues()::build)); 451 } 452 453 private static Document parseXmlDoc(String fileName, DocumentBuilder xmlDocBuilder) 454 throws SAXException, IOException { 455 try (InputStream configXml = new FileInputStream(new File(fileName))) { 456 Document xmlDoc = xmlDocBuilder.parse(configXml); 457 xmlDoc.getDocumentElement().normalize(); 458 return xmlDoc; 459 } 460 } 461 462 private static ImmutableList<Element> getElementsByTagName(Document xmlDoc, String tagName) { 463 if (xmlDoc == null) { 464 return ImmutableList.of(); 465 } 466 ImmutableList.Builder<Element> result = new ImmutableList.Builder<>(); 467 xmlDoc.getDocumentElement().normalize(); 468 NodeList nodeList = xmlDoc.getElementsByTagName(tagName); 469 for (int i = 0; i < nodeList.getLength(); i++) { 470 Node node = nodeList.item(i); 471 if (node.getNodeType() == Node.ELEMENT_NODE) { 472 result.add((Element) node); 473 } 474 } 475 return result.build(); 476 } 477 478 static CarrierConfig sortConfig(CarrierConfig in) { 479 final CarrierConfig.Builder result = in.toBuilder().clearConfig(); 480 in.getConfigList().stream() 481 .sorted(comparing(CarrierConfig.Config::getKey)) 482 .forEachOrdered((c) -> result.addConfig(c)); 483 return result.build(); 484 } 485 486 static String getCanonicalName(CarrierList pList, CarrierId pId) { 487 for (int i = 0; i < pList.getEntryCount(); i++) { 488 CarrierMap cm = pList.getEntry(i); 489 for (int j = 0; j < cm.getCarrierIdCount(); j++) { 490 CarrierId cid = cm.getCarrierId(j); 491 if (cid.equals(pId)) { 492 return cm.getCanonicalName(); 493 } 494 } 495 } 496 return null; 497 } 498 499 static String generateCanonicalNameForOthers(CarrierId pId) { 500 // Not a tier-1 carrier: generate name 501 StringBuilder genName = new StringBuilder(pId.getMccMnc()); 502 switch (pId.getMvnoDataCase()) { 503 case GID1: 504 genName.append("GID1="); 505 genName.append(Ascii.toUpperCase(pId.getGid1())); 506 break; 507 case SPN: 508 genName.append("SPN="); 509 genName.append(Ascii.toUpperCase(pId.getSpn())); 510 break; 511 case IMSI: 512 genName.append("IMSI="); 513 genName.append(Ascii.toUpperCase(pId.getImsi())); 514 break; 515 default: // MVNODATA_NOT_SET 516 // Do nothing 517 } 518 return genName.toString(); 519 } 520 521 /** 522 * Converts a map with carrier configs to a {@link CarrierConfig.Builder}. 523 * 524 * @see #parseCarrierConfigToMap 525 */ 526 private static CarrierConfig.Builder toCarrierConfigBuilder( 527 Map<String, CarrierConfig.Config> configs) { 528 CarrierConfig.Builder builder = CarrierConfig.newBuilder(); 529 configs.forEach( 530 (key, value) -> { 531 builder.addConfig(value.toBuilder().setKey(key)); 532 }); 533 return builder; 534 } 535 536 /** 537 * Returns a map with carrier configs parsed from a assets/*.xml. 538 * 539 * @return a map, key being the carrier config key, value being a {@link CarrierConfig.Config} 540 * with one of the value set. 541 */ 542 private static HashMap<String, CarrierConfig.Config> parseCarrierConfigFromXml( 543 Document xmlDoc, CarrierIdentifier carrier) throws IOException { 544 HashMap<String, CarrierConfig.Config> configMap = new HashMap<>(); 545 for (Element element : getElementsByTagName(xmlDoc, TAG_CARRIER_CONFIG)) { 546 if (carrier != null && !checkFilters(element, carrier)) { 547 continue; 548 } 549 configMap.putAll(parseCarrierConfigToMap(element)); 550 } 551 return configMap; 552 } 553 554 /** 555 * Returns a map with carrier configs parsed from the vendor.xml. 556 * 557 * @return a map, key being the carrier config key, value being a {@link CarrierConfig.Config} 558 * with one of the value set. 559 */ 560 private static HashMap<String, CarrierConfig.Config> parseCarrierConfigFromVendorXml( 561 Document xmlDoc, CarrierIdentifier carrier) throws IOException { 562 HashMap<String, CarrierConfig.Config> configMap = new HashMap<>(); 563 for (Element element : getElementsByTagName(xmlDoc, TAG_CARRIER_CONFIG)) { 564 if (carrier != null && !checkFilters(element, carrier)) { 565 continue; 566 } 567 configMap.putAll(parseCarrierConfigToMap(element)); 568 } 569 return configMap; 570 } 571 572 /** 573 * Returns a map with carrier configs parsed from the XML element. 574 * 575 * @return a map, key being the carrier config key, value being a {@link CarrierConfig.Config} 576 * with one of the value set. 577 */ 578 private static HashMap<String, CarrierConfig.Config> parseCarrierConfigToMap(Element element) 579 throws IOException { 580 HashMap<String, CarrierConfig.Config> configMap = new HashMap<>(); 581 NodeList nList; 582 // bool value 583 nList = element.getElementsByTagName("boolean"); 584 for (int i = 0; i < nList.getLength(); i++) { 585 Node nNode = nList.item(i); 586 if (nNode.getNodeType() != Node.ELEMENT_NODE) { 587 continue; 588 } 589 Element eElement = (Element) nNode; 590 String key = eElement.getAttribute("name"); 591 boolean value = Boolean.parseBoolean(eElement.getAttribute("value")); 592 configMap.put(key, CarrierConfig.Config.newBuilder().setBoolValue(value).build()); 593 } 594 // int value 595 nList = element.getElementsByTagName("int"); 596 for (int i = 0; i < nList.getLength(); i++) { 597 Node nNode = nList.item(i); 598 if (nNode.getNodeType() != Node.ELEMENT_NODE) { 599 continue; 600 } 601 Element eElement = (Element) nNode; 602 String key = eElement.getAttribute("name"); 603 int value = Integer.parseInt(eElement.getAttribute("value")); 604 configMap.put(key, CarrierConfig.Config.newBuilder().setIntValue(value).build()); 605 } 606 // long value 607 nList = element.getElementsByTagName("long"); 608 for (int i = 0; i < nList.getLength(); i++) { 609 Node nNode = nList.item(i); 610 if (nNode.getNodeType() != Node.ELEMENT_NODE) { 611 continue; 612 } 613 Element eElement = (Element) nNode; 614 String key = eElement.getAttribute("name"); 615 long value = Long.parseLong(eElement.getAttribute("value")); 616 configMap.put(key, CarrierConfig.Config.newBuilder().setLongValue(value).build()); 617 } 618 // text value 619 nList = element.getElementsByTagName("string"); 620 for (int i = 0; i < nList.getLength(); i++) { 621 Node nNode = nList.item(i); 622 if (nNode.getNodeType() != Node.ELEMENT_NODE) { 623 continue; 624 } 625 Element eElement = (Element) nNode; 626 String key = eElement.getAttribute("name"); 627 String value = String.valueOf(eElement.getTextContent()); 628 if (value.isEmpty()) { 629 value = eElement.getAttribute("value"); 630 } 631 configMap.put(key, CarrierConfig.Config.newBuilder().setTextValue(value).build()); 632 } 633 // text array 634 nList = element.getElementsByTagName("string-array"); 635 for (int i = 0; i < nList.getLength(); i++) { 636 Node nNode = nList.item(i); 637 if (nNode.getNodeType() != Node.ELEMENT_NODE) { 638 continue; 639 } 640 Element eElement = (Element) nNode; 641 String key = eElement.getAttribute("name"); 642 CarrierConfig.Config.Builder cccb = CarrierConfig.Config.newBuilder(); 643 TextArray.Builder cctb = TextArray.newBuilder(); 644 NodeList subList = eElement.getElementsByTagName("item"); 645 for (int j = 0; j < subList.getLength(); j++) { 646 Node subNode = subList.item(j); 647 if (subNode.getNodeType() != Node.ELEMENT_NODE) { 648 continue; 649 } 650 Element subElement = (Element) subNode; 651 String value = String.valueOf(subElement.getAttribute("value")); 652 cctb.addItem(value); 653 } 654 configMap.put(key, cccb.setTextArray(cctb.build()).build()); 655 } 656 // bool array 657 nList = element.getElementsByTagName("int-array"); 658 for (int i = 0; i < nList.getLength(); i++) { 659 Node nNode = nList.item(i); 660 if (nNode.getNodeType() != Node.ELEMENT_NODE) { 661 continue; 662 } 663 Element eElement = (Element) nNode; 664 String key = eElement.getAttribute("name"); 665 CarrierConfig.Config.Builder cccb = CarrierConfig.Config.newBuilder(); 666 IntArray.Builder ccib = IntArray.newBuilder(); 667 NodeList subList = eElement.getElementsByTagName("item"); 668 for (int j = 0; j < subList.getLength(); j++) { 669 Node subNode = subList.item(j); 670 if (subNode.getNodeType() != Node.ELEMENT_NODE) { 671 continue; 672 } 673 Element subElement = (Element) subNode; 674 int value = Integer.parseInt(subElement.getAttribute("value")); 675 ccib.addItem(value); 676 } 677 configMap.put(key, cccb.setIntArray(ccib.build()).build()); 678 } 679 return configMap; 680 } 681 682 /** 683 * Returns {@code true} if a <carrier_config ...> element matches the carrier identifier. 684 * 685 * <p>Copied from AOSP DefaultCarrierConfigService. 686 */ 687 private static boolean checkFilters(Element element, CarrierIdentifier id) { 688 boolean result = true; 689 NamedNodeMap attributes = element.getAttributes(); 690 for (int i = 0; i < attributes.getLength(); i++) { 691 String attribute = attributes.item(i).getNodeName(); 692 String value = attributes.item(i).getNodeValue(); 693 switch (attribute) { 694 case "mcc": 695 result = result && value.equals(id.getMcc()); 696 break; 697 case "mnc": 698 result = result && value.equals(id.getMnc()); 699 break; 700 case "gid1": 701 result = result && Ascii.equalsIgnoreCase(value, id.getGid1()); 702 break; 703 case "spn": 704 result = result && matchOnSP(value, id); 705 break; 706 case "imsi": 707 result = result && matchOnImsi(value, id); 708 break; 709 case "cid": 710 result = 711 result 712 && ((Integer.parseInt(value) == id.getCarrierId()) 713 || (Integer.parseInt(value) == id.getSpecificCarrierId())); 714 break; 715 case "name": 716 // name is used together with cid for readability. ignore for filter. 717 break; 718 default: 719 System.err.println("Unsupported attribute " + attribute + "=" + value); 720 result = false; 721 } 722 } 723 return result; 724 } 725 726 /** 727 * Returns {@code true} if an "spn" attribute in <carrier_config ...> element matches the carrier 728 * identifier. 729 * 730 * <p>Copied from AOSP DefaultCarrierConfigService. 731 */ 732 private static boolean matchOnSP(String xmlSP, CarrierIdentifier id) { 733 boolean matchFound = false; 734 735 String currentSP = id.getSpn(); 736 // <carrier_config ... spn="null"> means expecting SIM SPN empty in AOSP convention. 737 if (Ascii.equalsIgnoreCase("null", xmlSP)) { 738 if (currentSP.isEmpty()) { 739 matchFound = true; 740 } 741 } else if (currentSP != null) { 742 Pattern spPattern = Pattern.compile(xmlSP, Pattern.CASE_INSENSITIVE); 743 Matcher matcher = spPattern.matcher(currentSP); 744 matchFound = matcher.matches(); 745 } 746 return matchFound; 747 } 748 749 /** 750 * Returns {@code true} if an "imsi" attribute in <carrier_config ...> element matches the carrier 751 * identifier. 752 * 753 * <p>Copied from AOSP DefaultCarrierConfigService. 754 */ 755 private static boolean matchOnImsi(String xmlImsi, CarrierIdentifier id) { 756 boolean matchFound = false; 757 758 String currentImsi = id.getImsi(); 759 // If we were able to retrieve current IMSI, see if it matches. 760 if (currentImsi != null) { 761 Pattern imsiPattern = Pattern.compile(xmlImsi, Pattern.CASE_INSENSITIVE); 762 Matcher matcher = imsiPattern.matcher(currentImsi); 763 matchFound = matcher.matches(); 764 } 765 return matchFound; 766 } 767 768 /** 769 * Parses a {@link CarrierId} out of a <carrier_config ...> tag. 770 * 771 * <p>This is purely used for discover potential carriers expressed by this tag, the return value 772 * may not reflect all attributes of the tag. 773 */ 774 private static CarrierId.Builder parseCarrierId(Element element) { 775 CarrierId.Builder builder = CarrierId.newBuilder(); 776 String mccMnc = element.getAttribute("mcc") + element.getAttribute("mnc"); 777 builder.setMccMnc(mccMnc); 778 if (element.hasAttribute("imsi")) { 779 builder.setImsi(element.getAttribute("imsi")); 780 } else if (element.hasAttribute("gid1")) { 781 builder.setGid1(element.getAttribute("gid1")); 782 } else if (element.hasAttribute("gid2")) { 783 throw new UnsupportedOperationException( 784 "Not support attribute `gid2`: " + element.getAttribute("gid2")); 785 } else if (element.hasAttribute("spn")) { 786 builder.setSpn(element.getAttribute("spn")); 787 } 788 return builder; 789 } 790 791 // Same as {@link java.nio.file.Paths#get} but returns a String 792 private static String getPathAsString(String first, String... more) { 793 return java.nio.file.Paths.get(first, more).toString(); 794 } 795 796 /** Mirror of Android CarrierIdentifier class. Default value of a carrier id is -1. */ 797 @AutoValue 798 abstract static class CarrierIdentifier { 799 abstract String getMcc(); 800 801 abstract String getMnc(); 802 803 abstract String getImsi(); 804 805 abstract String getGid1(); 806 807 abstract String getSpn(); 808 809 abstract int getCarrierId(); 810 811 abstract int getSpecificCarrierId(); 812 813 abstract int getMccmncCarrierId(); 814 815 static CarrierIdentifier create( 816 CarrierId carrier, int carrierId, int specificCarrierId, int mccmncCarrierId) { 817 String mcc = carrier.getMccMnc().substring(0, 3); 818 String mnc = carrier.getMccMnc().length() > 3 ? carrier.getMccMnc().substring(3) : ""; 819 return new AutoValue_CarrierConfigConverterV2_CarrierIdentifier( 820 mcc, 821 mnc, 822 carrier.getImsi(), 823 carrier.getGid1(), 824 carrier.getSpn(), 825 carrierId, 826 specificCarrierId, 827 mccmncCarrierId); 828 } 829 } 830 831 private static CarrierIdentifier getCid( 832 CarrierId carrierId, Multimap<CarrierId, Integer> reverseAospCarrierList) { 833 // Mimic TelephonyManager#getCarrierIdFromMccMnc, which is implemented by 834 // CarrierResolver#getCarrierIdFromMccMnc. 835 CarrierId mccMnc = CarrierId.newBuilder().setMccMnc(carrierId.getMccMnc()).build(); 836 int mccMncCarrierId = reverseAospCarrierList.get(mccMnc).stream().findFirst().orElse(-1); 837 838 List<Integer> cids = ImmutableList.copyOf(reverseAospCarrierList.get(carrierId)); 839 // No match: use -1 840 if (cids.isEmpty()) { 841 return CarrierIdentifier.create(carrierId, -1, -1, mccMncCarrierId); 842 } 843 // One match: use as both carrierId and specificCarrierId 844 if (cids.size() == 1) { 845 return CarrierIdentifier.create(carrierId, cids.get(0), cids.get(0), mccMncCarrierId); 846 } 847 // Two matches: specificCarrierId is always bigger than carrierId 848 if (cids.size() == 2) { 849 return CarrierIdentifier.create( 850 carrierId, 851 Math.min(cids.get(0), cids.get(1)), 852 Math.max(cids.get(0), cids.get(1)), 853 mccMncCarrierId); 854 } 855 // Cannot be more than 2 matches. 856 throw new IllegalStateException("More than two cid's found for " + carrierId + ": " + cids); 857 } 858 859 private CarrierConfigConverterV2() {} 860 } 861