Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 1 | // Copyright (c) 2023, the R8 project authors. Please see the AUTHORS file |
| 2 | // for details. All rights reserved. Use of this source code is governed by a |
| 3 | // BSD-style license that can be found in the LICENSE file. |
| 4 | |
| 5 | package com.android.build.shrinker.r8integration; |
| 6 | |
| 7 | import static java.nio.charset.StandardCharsets.UTF_16BE; |
| 8 | import static java.nio.charset.StandardCharsets.UTF_16LE; |
| 9 | import static java.nio.charset.StandardCharsets.UTF_8; |
| 10 | |
| 11 | import com.android.aapt.Resources.ResourceTable; |
| 12 | import com.android.aapt.Resources.XmlNode; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 13 | import com.android.build.shrinker.ResourceShrinkerImplKt; |
| 14 | import com.android.build.shrinker.ResourceTableUtilKt; |
Rico Wind | c5b4401 | 2024-08-16 13:30:59 +0200 | [diff] [blame] | 15 | import com.android.build.shrinker.ShrinkerDebugReporter; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 16 | import com.android.build.shrinker.graph.ProtoResourcesGraphBuilder; |
Rico Wind | 7eaee65 | 2024-01-05 09:47:54 +0100 | [diff] [blame] | 17 | import com.android.build.shrinker.obfuscation.ProguardMappingsRecorder; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 18 | import com.android.build.shrinker.r8integration.R8ResourceShrinkerState.R8ResourceShrinkerModel; |
| 19 | import com.android.build.shrinker.usages.DexFileAnalysisCallback; |
| 20 | import com.android.build.shrinker.usages.ProtoAndroidManifestUsageRecorderKt; |
| 21 | import com.android.build.shrinker.usages.R8ResourceShrinker; |
| 22 | import com.android.build.shrinker.usages.ToolsAttributeUsageRecorderKt; |
Rico Wind | c25e07e | 2023-10-30 08:41:12 +0100 | [diff] [blame] | 23 | import com.android.ide.common.resources.ResourcesUtil; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 24 | import com.android.ide.common.resources.usage.ResourceStore; |
Rico Wind | ee46512 | 2024-01-15 09:41:09 +0100 | [diff] [blame] | 25 | import com.android.ide.common.resources.usage.ResourceUsageModel.Resource; |
Rico Wind | 26aaf90 | 2024-01-29 12:59:53 +0100 | [diff] [blame] | 26 | import com.android.resources.ResourceType; |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 27 | import com.android.tools.r8.FeatureSplit; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 28 | import com.google.common.collect.ImmutableMap; |
| 29 | import com.google.common.collect.ImmutableSet; |
| 30 | import com.google.common.collect.Iterables; |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 31 | import com.google.protobuf.InvalidProtocolBufferException; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 32 | import java.io.ByteArrayInputStream; |
| 33 | import java.io.IOException; |
| 34 | import java.io.InputStreamReader; |
| 35 | import java.io.Reader; |
| 36 | import java.io.StringReader; |
| 37 | import java.nio.file.Path; |
| 38 | import java.nio.file.Paths; |
| 39 | import java.util.ArrayList; |
Rico Wind | 5b0579f | 2024-08-13 08:46:59 +0200 | [diff] [blame] | 40 | import java.util.Collection; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 41 | import java.util.HashMap; |
| 42 | import java.util.List; |
| 43 | import java.util.Map; |
| 44 | import java.util.Map.Entry; |
| 45 | import java.util.Set; |
| 46 | import java.util.stream.Collectors; |
| 47 | import javax.xml.parsers.ParserConfigurationException; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 48 | import org.xml.sax.SAXException; |
| 49 | |
| 50 | public class LegacyResourceShrinker { |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 51 | private final Map<String, byte[]> dexInputs; |
Rico Wind | 5b0579f | 2024-08-13 08:46:59 +0200 | [diff] [blame] | 52 | private final Collection<PathAndBytes> resFolderInputs; |
| 53 | private final Collection<PathAndBytes> xmlInputs; |
Rico Wind | fd460d2 | 2024-08-20 11:06:18 +0200 | [diff] [blame] | 54 | private final List<byte[]> keepRuleInput; |
Rico Wind | 7eaee65 | 2024-01-05 09:47:54 +0100 | [diff] [blame] | 55 | private List<String> proguardMapStrings; |
Rico Wind | c5b4401 | 2024-08-16 13:30:59 +0200 | [diff] [blame] | 56 | private final ShrinkerDebugReporter debugReporter; |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 57 | private final List<PathAndBytes> manifest; |
| 58 | private final Map<PathAndBytes, FeatureSplit> resourceTables; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 59 | |
| 60 | public static class Builder { |
| 61 | |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 62 | private final Map<String, byte[]> dexInputs = new HashMap<>(); |
Rico Wind | 5d22f0d | 2024-09-19 13:23:04 +0200 | [diff] [blame] | 63 | private final Map<String, PathAndBytes> resFolderInputs = new HashMap<>(); |
| 64 | private final Map<String, PathAndBytes> xmlInputs = new HashMap<>(); |
Rico Wind | fd460d2 | 2024-08-20 11:06:18 +0200 | [diff] [blame] | 65 | private final List<byte[]> keepRuleInput = new ArrayList<>(); |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 66 | |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 67 | private final List<PathAndBytes> manifests = new ArrayList<>(); |
| 68 | private final Map<PathAndBytes, FeatureSplit> resourceTables = new HashMap<>(); |
Rico Wind | 7eaee65 | 2024-01-05 09:47:54 +0100 | [diff] [blame] | 69 | private List<String> proguardMapStrings; |
Rico Wind | c5b4401 | 2024-08-16 13:30:59 +0200 | [diff] [blame] | 70 | private ShrinkerDebugReporter debugReporter; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 71 | |
| 72 | private Builder() {} |
| 73 | |
Rico Wind | 5d22f0d | 2024-09-19 13:23:04 +0200 | [diff] [blame] | 74 | public Builder addManifest(String path, byte[] bytes) { |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 75 | manifests.add(new PathAndBytes(bytes, path)); |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 76 | return this; |
| 77 | } |
| 78 | |
Rico Wind | 5d22f0d | 2024-09-19 13:23:04 +0200 | [diff] [blame] | 79 | public Builder addResourceTable(String path, byte[] bytes, FeatureSplit featureSplit) { |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 80 | resourceTables.put(new PathAndBytes(bytes, path), featureSplit); |
| 81 | try { |
| 82 | ResourceTable resourceTable = ResourceTable.parseFrom(bytes); |
| 83 | System.currentTimeMillis(); |
| 84 | } catch (InvalidProtocolBufferException e) { |
| 85 | throw new RuntimeException(e); |
| 86 | } |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 87 | return this; |
| 88 | } |
| 89 | |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 90 | public Builder addDexInput(String classesLocation, byte[] bytes) { |
| 91 | dexInputs.put(classesLocation, bytes); |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 92 | return this; |
| 93 | } |
| 94 | |
Rico Wind | fd460d2 | 2024-08-20 11:06:18 +0200 | [diff] [blame] | 95 | public Builder addKeepRuleInput(byte[] bytes) { |
| 96 | keepRuleInput.add(bytes); |
| 97 | return this; |
| 98 | } |
| 99 | |
Rico Wind | 5d22f0d | 2024-09-19 13:23:04 +0200 | [diff] [blame] | 100 | public Builder addResFolderInput(String path, byte[] bytes) { |
Rico Wind | 5b0579f | 2024-08-13 08:46:59 +0200 | [diff] [blame] | 101 | PathAndBytes existing = resFolderInputs.get(path); |
| 102 | if (existing != null) { |
Rico Wind | 32420ca | 2024-12-19 15:04:52 +0100 | [diff] [blame] | 103 | existing.setDuplicated(true); |
Rico Wind | 5b0579f | 2024-08-13 08:46:59 +0200 | [diff] [blame] | 104 | } else { |
| 105 | resFolderInputs.put(path, new PathAndBytes(bytes, path)); |
| 106 | } |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 107 | return this; |
| 108 | } |
| 109 | |
Rico Wind | 5d22f0d | 2024-09-19 13:23:04 +0200 | [diff] [blame] | 110 | public Builder addXmlInput(String path, byte[] bytes) { |
Rico Wind | 5b0579f | 2024-08-13 08:46:59 +0200 | [diff] [blame] | 111 | PathAndBytes existing = xmlInputs.get(path); |
| 112 | if (existing != null) { |
Rico Wind | 32420ca | 2024-12-19 15:04:52 +0100 | [diff] [blame] | 113 | existing.setDuplicated(true); |
Rico Wind | 5b0579f | 2024-08-13 08:46:59 +0200 | [diff] [blame] | 114 | } else { |
| 115 | xmlInputs.put(path, new PathAndBytes(bytes, path)); |
| 116 | } |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 117 | return this; |
| 118 | } |
| 119 | |
| 120 | public LegacyResourceShrinker build() { |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 121 | assert manifests != null && resourceTables != null; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 122 | return new LegacyResourceShrinker( |
Rico Wind | 5b0579f | 2024-08-13 08:46:59 +0200 | [diff] [blame] | 123 | dexInputs, |
| 124 | resFolderInputs.values(), |
| 125 | manifests, |
| 126 | resourceTables, |
| 127 | xmlInputs.values(), |
Rico Wind | fd460d2 | 2024-08-20 11:06:18 +0200 | [diff] [blame] | 128 | keepRuleInput, |
Rico Wind | c5b4401 | 2024-08-16 13:30:59 +0200 | [diff] [blame] | 129 | proguardMapStrings, |
| 130 | debugReporter); |
Rico Wind | 7eaee65 | 2024-01-05 09:47:54 +0100 | [diff] [blame] | 131 | } |
| 132 | |
| 133 | public void setProguardMapStrings(List<String> proguardMapStrings) { |
| 134 | this.proguardMapStrings = proguardMapStrings; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 135 | } |
Rico Wind | c5b4401 | 2024-08-16 13:30:59 +0200 | [diff] [blame] | 136 | |
| 137 | public Builder setShrinkerDebugReporter(ShrinkerDebugReporter debugReporter) { |
| 138 | this.debugReporter = debugReporter; |
| 139 | return this; |
| 140 | } |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 141 | } |
| 142 | |
| 143 | private LegacyResourceShrinker( |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 144 | Map<String, byte[]> dexInputs, |
Rico Wind | 5b0579f | 2024-08-13 08:46:59 +0200 | [diff] [blame] | 145 | Collection<PathAndBytes> resFolderInputs, |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 146 | List<PathAndBytes> manifests, |
| 147 | Map<PathAndBytes, FeatureSplit> resourceTables, |
Rico Wind | 5b0579f | 2024-08-13 08:46:59 +0200 | [diff] [blame] | 148 | Collection<PathAndBytes> xmlInputs, |
Rico Wind | fd460d2 | 2024-08-20 11:06:18 +0200 | [diff] [blame] | 149 | List<byte[]> additionalRawXmlInputs, |
Rico Wind | c5b4401 | 2024-08-16 13:30:59 +0200 | [diff] [blame] | 150 | List<String> proguardMapStrings, |
| 151 | ShrinkerDebugReporter debugReporter) { |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 152 | this.dexInputs = dexInputs; |
| 153 | this.resFolderInputs = resFolderInputs; |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 154 | this.manifest = manifests; |
| 155 | this.resourceTables = resourceTables; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 156 | this.xmlInputs = xmlInputs; |
Rico Wind | fd460d2 | 2024-08-20 11:06:18 +0200 | [diff] [blame] | 157 | this.keepRuleInput = additionalRawXmlInputs; |
Rico Wind | 7eaee65 | 2024-01-05 09:47:54 +0100 | [diff] [blame] | 158 | this.proguardMapStrings = proguardMapStrings; |
Rico Wind | c5b4401 | 2024-08-16 13:30:59 +0200 | [diff] [blame] | 159 | this.debugReporter = debugReporter; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 160 | } |
| 161 | |
| 162 | public static Builder builder() { |
| 163 | return new Builder(); |
| 164 | } |
| 165 | |
| 166 | public ShrinkerResult run() throws IOException, ParserConfigurationException, SAXException { |
Rico Wind | c5b4401 | 2024-08-16 13:30:59 +0200 | [diff] [blame] | 167 | R8ResourceShrinkerModel model = new R8ResourceShrinkerModel(debugReporter, true); |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 168 | for (PathAndBytes pathAndBytes : resourceTables.keySet()) { |
| 169 | ResourceTable loadedResourceTable = ResourceTable.parseFrom(pathAndBytes.bytes); |
Rico Wind | ba8a9a4 | 2024-01-15 11:22:01 +0100 | [diff] [blame] | 170 | model.instantiateFromResourceTable(loadedResourceTable, false); |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 171 | } |
Rico Wind | 7eaee65 | 2024-01-05 09:47:54 +0100 | [diff] [blame] | 172 | if (proguardMapStrings != null) { |
| 173 | new ProguardMappingsRecorder(proguardMapStrings).recordObfuscationMappings(model); |
| 174 | proguardMapStrings = null; |
| 175 | } |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 176 | for (Entry<String, byte[]> entry : dexInputs.entrySet()) { |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 177 | // The analysis needs an origin for the dex files, synthesize an easy recognizable one. |
Rico Wind | c5b4401 | 2024-08-16 13:30:59 +0200 | [diff] [blame] | 178 | Path inMemoryR8 = Paths.get("in_memory_r8_" + entry.getKey()); |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 179 | R8ResourceShrinker.runResourceShrinkerAnalysis( |
| 180 | entry.getValue(), inMemoryR8, new DexFileAnalysisCallback(inMemoryR8, model)); |
| 181 | } |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 182 | for (PathAndBytes pathAndBytes : manifest) { |
| 183 | ProtoAndroidManifestUsageRecorderKt.recordUsagesFromNode( |
| 184 | XmlNode.parseFrom(pathAndBytes.bytes), model); |
| 185 | } |
Rico Wind | fd460d2 | 2024-08-20 11:06:18 +0200 | [diff] [blame] | 186 | for (byte[] keepBytes : keepRuleInput) { |
| 187 | ToolsAttributeUsageRecorderKt.processRawXml(getUtfReader(keepBytes), model); |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 188 | } |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 189 | |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 190 | ImmutableMap<String, PathAndBytes> resFolderMappings = |
| 191 | new ImmutableMap.Builder<String, PathAndBytes>() |
| 192 | .putAll( |
| 193 | xmlInputs.stream() |
| 194 | .collect(Collectors.toMap(PathAndBytes::getPathWithoutRes, a -> a))) |
| 195 | .putAll( |
| 196 | resFolderInputs.stream() |
| 197 | .collect(Collectors.toMap(PathAndBytes::getPathWithoutRes, a -> a))) |
| 198 | .build(); |
| 199 | for (PathAndBytes pathAndBytes : resourceTables.keySet()) { |
| 200 | ResourceTable resourceTable = ResourceTable.parseFrom(pathAndBytes.bytes); |
| 201 | new ProtoResourcesGraphBuilder( |
Rico Wind | 717e973 | 2024-02-09 10:36:00 +0100 | [diff] [blame] | 202 | pathInRes -> resFolderMappings.get(pathInRes).getBytes(), unused -> resourceTable) |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 203 | .buildGraph(model); |
| 204 | } |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 205 | ResourceStore resourceStore = model.getResourceStore(); |
| 206 | resourceStore.processToolsAttributes(); |
| 207 | model.keepPossiblyReferencedResources(); |
Rico Wind | c5b4401 | 2024-08-16 13:30:59 +0200 | [diff] [blame] | 208 | debugReporter.debug(model.getResourceStore()::dumpResourceModel); |
Rico Wind | c25e07e | 2023-10-30 08:41:12 +0100 | [diff] [blame] | 209 | // Transitively mark the reachable resources in the model. |
| 210 | // Finds unused resources in provided resources collection. |
| 211 | // Marks all used resources as 'reachable' in original collection. |
Rico Wind | ee46512 | 2024-01-15 09:41:09 +0100 | [diff] [blame] | 212 | List<Resource> unusedResources = |
Rico Wind | dc5cae8 | 2025-03-10 17:56:18 +0100 | [diff] [blame] | 213 | new ArrayList<>( |
| 214 | ResourcesUtil.findUnusedResources( |
| 215 | model.getResourceStore().getResources(), |
| 216 | roots -> { |
| 217 | debugReporter.debug(() -> "The root reachable resources are:"); |
| 218 | roots.forEach(root -> debugReporter.debug(() -> " " + root)); |
| 219 | })); |
Rico Wind | 32420ca | 2024-12-19 15:04:52 +0100 | [diff] [blame] | 220 | ImmutableSet.Builder<String> resEntriesToKeepBuilder = new ImmutableSet.Builder<>(); |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 221 | for (PathAndBytes xmlInput : Iterables.concat(xmlInputs, resFolderInputs)) { |
| 222 | if (ResourceShrinkerImplKt.isJarPathReachable(resourceStore, xmlInput.path.toString())) { |
Rico Wind | 32420ca | 2024-12-19 15:04:52 +0100 | [diff] [blame] | 223 | resEntriesToKeepBuilder.add(xmlInput.path.toString()); |
| 224 | if (xmlInput.duplicated) { |
| 225 | // Ensure that we don't remove references to duplicated res folder entries. |
| 226 | List<Resource> duplicatedResources = |
| 227 | ResourceShrinkerImplKt.getResourcesFor(resourceStore, xmlInput.path.toString()); |
| 228 | unusedResources.removeAll(duplicatedResources); |
| 229 | } |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 230 | } |
| 231 | } |
Rico Wind | 32420ca | 2024-12-19 15:04:52 +0100 | [diff] [blame] | 232 | debugReporter.debug(() -> "Unused resources are: "); |
| 233 | unusedResources.forEach(unused -> debugReporter.debug(() -> " " + unused)); |
Rico Wind | 717e973 | 2024-02-09 10:36:00 +0100 | [diff] [blame] | 234 | List<Integer> resourceIdsToRemove = getResourceIdsToRemove(unusedResources); |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 235 | Map<FeatureSplit, ResourceTable> shrunkenTables = new HashMap<>(); |
| 236 | for (Entry<PathAndBytes, FeatureSplit> entry : resourceTables.entrySet()) { |
| 237 | ResourceTable shrunkenResourceTable = |
| 238 | ResourceTableUtilKt.nullOutEntriesWithIds( |
| 239 | ResourceTable.parseFrom(entry.getKey().bytes), resourceIdsToRemove); |
| 240 | shrunkenTables.put(entry.getValue(), shrunkenResourceTable); |
| 241 | } |
Rico Wind | 32420ca | 2024-12-19 15:04:52 +0100 | [diff] [blame] | 242 | return new ShrinkerResult(resEntriesToKeepBuilder.build(), shrunkenTables); |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 243 | } |
| 244 | |
Rico Wind | 717e973 | 2024-02-09 10:36:00 +0100 | [diff] [blame] | 245 | private static List<Integer> getResourceIdsToRemove(List<Resource> unusedResources) { |
Rico Wind | 26aaf90 | 2024-01-29 12:59:53 +0100 | [diff] [blame] | 246 | return unusedResources.stream() |
| 247 | .filter(s -> s.type != ResourceType.ID) |
| 248 | .map(resource -> resource.value) |
| 249 | .collect(Collectors.toList()); |
Rico Wind | ba8a9a4 | 2024-01-15 11:22:01 +0100 | [diff] [blame] | 250 | } |
| 251 | |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 252 | // Lifted from com/android/utils/XmlUtils.java which we can't easily update internal dependency |
| 253 | // for. |
| 254 | /** |
| 255 | * Returns a character reader for the given bytes, which must be a UTF encoded file. |
| 256 | * |
| 257 | * <p>The reader does not need to be closed by the caller (because the file is read in full in one |
| 258 | * shot and the resulting array is then wrapped in a byte array input stream, which does not need |
| 259 | * to be closed.) |
| 260 | */ |
| 261 | public static Reader getUtfReader(byte[] bytes) throws IOException { |
| 262 | int length = bytes.length; |
| 263 | if (length == 0) { |
| 264 | return new StringReader(""); |
| 265 | } |
| 266 | |
| 267 | switch (bytes[0]) { |
| 268 | case (byte) 0xEF: |
| 269 | { |
| 270 | if (length >= 3 && bytes[1] == (byte) 0xBB && bytes[2] == (byte) 0xBF) { |
| 271 | // UTF-8 BOM: EF BB BF: Skip it |
| 272 | return new InputStreamReader(new ByteArrayInputStream(bytes, 3, length - 3), UTF_8); |
| 273 | } |
| 274 | break; |
| 275 | } |
| 276 | case (byte) 0xFE: |
| 277 | { |
| 278 | if (length >= 2 && bytes[1] == (byte) 0xFF) { |
| 279 | // UTF-16 Big Endian BOM: FE FF |
| 280 | return new InputStreamReader(new ByteArrayInputStream(bytes, 2, length - 2), UTF_16BE); |
| 281 | } |
| 282 | break; |
| 283 | } |
| 284 | case (byte) 0xFF: |
| 285 | { |
| 286 | if (length >= 2 && bytes[1] == (byte) 0xFE) { |
| 287 | if (length >= 4 && bytes[2] == (byte) 0x00 && bytes[3] == (byte) 0x00) { |
| 288 | // UTF-32 Little Endian BOM: FF FE 00 00 |
| 289 | return new InputStreamReader( |
| 290 | new ByteArrayInputStream(bytes, 4, length - 4), "UTF-32LE"); |
| 291 | } |
| 292 | |
| 293 | // UTF-16 Little Endian BOM: FF FE |
| 294 | return new InputStreamReader(new ByteArrayInputStream(bytes, 2, length - 2), UTF_16LE); |
| 295 | } |
| 296 | break; |
| 297 | } |
| 298 | case (byte) 0x00: |
| 299 | { |
| 300 | if (length >= 4 |
| 301 | && bytes[0] == (byte) 0x00 |
| 302 | && bytes[1] == (byte) 0x00 |
| 303 | && bytes[2] == (byte) 0xFE |
| 304 | && bytes[3] == (byte) 0xFF) { |
| 305 | // UTF-32 Big Endian BOM: 00 00 FE FF |
| 306 | return new InputStreamReader( |
| 307 | new ByteArrayInputStream(bytes, 4, length - 4), "UTF-32BE"); |
| 308 | } |
| 309 | break; |
| 310 | } |
| 311 | } |
| 312 | |
| 313 | // No byte order mark: Assume UTF-8 (where the BOM is optional). |
| 314 | return new InputStreamReader(new ByteArrayInputStream(bytes), UTF_8); |
| 315 | } |
| 316 | |
| 317 | public static class ShrinkerResult { |
| 318 | private final Set<String> resFolderEntriesToKeep; |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 319 | private final Map<FeatureSplit, ResourceTable> resourceTableInProtoFormat; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 320 | |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 321 | public ShrinkerResult( |
| 322 | Set<String> resFolderEntriesToKeep, |
| 323 | Map<FeatureSplit, ResourceTable> resourceTableInProtoFormat) { |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 324 | this.resFolderEntriesToKeep = resFolderEntriesToKeep; |
| 325 | this.resourceTableInProtoFormat = resourceTableInProtoFormat; |
| 326 | } |
| 327 | |
Rico Wind | cc52b7f | 2023-11-09 11:29:54 +0100 | [diff] [blame] | 328 | public byte[] getResourceTableInProtoFormat(FeatureSplit featureSplit) { |
| 329 | return resourceTableInProtoFormat.get(featureSplit).toByteArray(); |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 330 | } |
| 331 | |
| 332 | public Set<String> getResFolderEntriesToKeep() { |
| 333 | return resFolderEntriesToKeep; |
| 334 | } |
| 335 | } |
| 336 | |
| 337 | private static class PathAndBytes { |
| 338 | private final byte[] bytes; |
Rico Wind | 5d22f0d | 2024-09-19 13:23:04 +0200 | [diff] [blame] | 339 | private final String path; |
Rico Wind | 32420ca | 2024-12-19 15:04:52 +0100 | [diff] [blame] | 340 | private boolean duplicated; |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 341 | |
Rico Wind | 5d22f0d | 2024-09-19 13:23:04 +0200 | [diff] [blame] | 342 | private PathAndBytes(byte[] bytes, String path) { |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 343 | this.bytes = bytes; |
| 344 | this.path = path; |
| 345 | } |
| 346 | |
Rico Wind | 32420ca | 2024-12-19 15:04:52 +0100 | [diff] [blame] | 347 | public void setDuplicated(boolean duplicated) { |
| 348 | this.duplicated = duplicated; |
| 349 | } |
| 350 | |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 351 | public String getPathWithoutRes() { |
Rico Wind | 5d22f0d | 2024-09-19 13:23:04 +0200 | [diff] [blame] | 352 | assert path.startsWith("res/"); |
| 353 | return path.substring(4); |
Rico Wind | 441f03d | 2023-10-26 12:40:07 +0200 | [diff] [blame] | 354 | } |
| 355 | |
| 356 | public byte[] getBytes() { |
| 357 | return bytes; |
| 358 | } |
| 359 | } |
| 360 | } |