blob: 2b782959ede52a816d3012063fafffaf32f5f806 [file] [log] [blame]
// Copyright (c) 2016, the R8 project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
package com.android.tools.r8.utils;
import static com.android.tools.r8.utils.FileUtils.isArchive;
import static com.android.tools.r8.utils.FileUtils.isClassFile;
import static com.android.tools.r8.utils.FileUtils.isDexFile;
import com.android.tools.r8.Resource;
import com.android.tools.r8.ResourceProvider;
import com.android.tools.r8.errors.CompilationError;
import com.google.common.collect.ImmutableList;
import com.google.common.io.ByteStreams;
import com.google.common.io.Closer;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.charset.StandardCharsets;
import java.nio.file.CopyOption;
import java.nio.file.Files;
import java.nio.file.OpenOption;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardCopyOption;
import java.nio.file.StandardOpenOption;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Set;
import java.util.zip.ZipEntry;
import java.util.zip.ZipException;
import java.util.zip.ZipInputStream;
import java.util.zip.ZipOutputStream;
/**
* Collection of program files needed for processing.
*
* <p>This abstraction is the main input and output container for a given application.
*/
public class AndroidApp {
public static final String DEFAULT_PROGUARD_MAP_FILE = "proguard.map";
public static final String DEFAULT_PROGUARD_SEEDS_FILE = "proguard.seeds";
public static final String DEFAULT_PACKAGE_DISTRIBUTION_FILE = "package.map";
private final ImmutableList<InternalResource> dexSources;
private final ImmutableList<InternalResource> classSources;
private final ImmutableList<ResourceProvider> resourceProviders;
private final InternalResource proguardMap;
private final InternalResource proguardSeeds;
private final InternalResource packageDistribution;
private final InternalResource mainDexList;
// See factory methods and AndroidApp.Builder below.
private AndroidApp(
ImmutableList<InternalResource> dexSources,
ImmutableList<InternalResource> classSources,
ImmutableList<ResourceProvider> resourceProviders,
InternalResource proguardMap,
InternalResource proguardSeeds,
InternalResource packageDistribution,
InternalResource mainDexList) {
this.dexSources = dexSources;
this.classSources = classSources;
this.resourceProviders = resourceProviders;
this.proguardMap = proguardMap;
this.proguardSeeds = proguardSeeds;
this.packageDistribution = packageDistribution;
this.mainDexList = mainDexList;
}
/**
* Create a new empty builder.
*/
public static Builder builder() {
return new Builder();
}
/**
* Create a new builder initialized with the resources from @code{app}.
*/
public static Builder builder(AndroidApp app) {
return new Builder(app);
}
/**
* Create an app from program files @code{files}. See also Builder::addProgramFiles.
*/
public static AndroidApp fromProgramFiles(Path... files) throws IOException {
return fromProgramFiles(Arrays.asList(files));
}
/**
* Create an app from program files @code{files}. See also Builder::addProgramFiles.
*/
public static AndroidApp fromProgramFiles(List<Path> files) throws IOException {
return builder().addProgramFiles(files).build();
}
/**
* Create an app from files found in @code{directory}. See also Builder::addProgramDirectory.
*/
public static AndroidApp fromProgramDirectory(Path directory) throws IOException {
return builder().addProgramDirectory(directory).build();
}
/**
* Create an app from dex program data. See also Builder::addDexProgramData.
*/
public static AndroidApp fromDexProgramData(byte[]... data) {
return fromDexProgramData(Arrays.asList(data));
}
/**
* Create an app from dex program data. See also Builder::addDexProgramData.
*/
public static AndroidApp fromDexProgramData(List<byte[]> data) {
return builder().addDexProgramData(data).build();
}
/**
* Create an app from Java-bytecode program data. See also Builder::addClassProgramData.
*/
public static AndroidApp fromClassProgramData(byte[]... data) {
return fromClassProgramData(Arrays.asList(data));
}
/**
* Create an app from Java-bytecode program data. See also Builder::addClassProgramData.
*/
public static AndroidApp fromClassProgramData(List<byte[]> data) {
return builder().addClassProgramData(data).build();
}
/** Get input streams for all dex program resources. */
public List<InternalResource> getDexProgramResources() {
return filter(dexSources, Resource.Kind.PROGRAM);
}
/** Get input streams for all Java-bytecode program resources. */
public List<InternalResource> getClassProgramResources() {
return filter(classSources, Resource.Kind.PROGRAM);
}
/** Get input streams for all dex program classpath resources. */
public List<InternalResource> getDexClasspathResources() {
return filter(dexSources, Resource.Kind.CLASSPATH);
}
/** Get input streams for all Java-bytecode classpath resources. */
public List<InternalResource> getClassClasspathResources() {
return filter(classSources, Resource.Kind.CLASSPATH);
}
/** Get input streams for all dex library resources. */
public List<InternalResource> getDexLibraryResources() {
return filter(dexSources, Resource.Kind.LIBRARY);
}
/** Get input streams for all Java-bytecode library resources. */
public List<InternalResource> getClassLibraryResources() {
return filter(classSources, Resource.Kind.LIBRARY);
}
/** Get lazy resource providers. */
public List<ResourceProvider> getLazyResourceProviders() {
return resourceProviders;
}
private List<InternalResource> filter(
List<InternalResource> resources, Resource.Kind kind) {
List<InternalResource> out = new ArrayList<>(resources.size());
for (InternalResource resource : resources) {
if (kind == resource.getKind()) {
out.add(resource);
}
}
return out;
}
/**
* True if the proguard-map resource exists.
*/
public boolean hasProguardMap() {
return proguardMap != null;
}
/**
* Get the input stream of the proguard-map resource if it exists.
*/
public InputStream getProguardMap(Closer closer) throws IOException {
return proguardMap == null ? null : proguardMap.getStream(closer);
}
/**
* True if the proguard-seeds resource exists.
*/
public boolean hasProguardSeeds() {
return proguardSeeds != null;
}
/**
* Get the input stream of the proguard-seeds resource if it exists.
*/
public InputStream getProguardSeeds(Closer closer) throws IOException {
return proguardSeeds == null ? null : proguardSeeds.getStream(closer);
}
/**
* True if the package distribution resource exists.
*/
public boolean hasPackageDistribution() {
return packageDistribution != null;
}
/**
* Get the input stream of the package distribution resource if it exists.
*/
public InputStream getPackageDistribution(Closer closer) throws IOException {
return packageDistribution == null ? null : packageDistribution.getStream(closer);
}
/**
* True if the main dex list resource exists.
*/
public boolean hasMainDexList() {
return mainDexList != null;
}
/**
* Get the input stream of the main dex list resource if it exists.
*/
public InputStream getMainDexList(Closer closer) throws IOException {
return mainDexList == null ? null : mainDexList.getStream(closer);
}
/**
* Write the dex program resources and proguard resource to @code{output}.
*/
public void write(Path output, OutputMode outputMode) throws IOException {
write(output, outputMode, false);
}
/**
* Write the dex program resources and proguard resource to @code{output}.
*/
public void write(Path output, OutputMode outputMode, boolean overwrite) throws IOException {
if (isArchive(output)) {
writeToZip(output, outputMode, overwrite);
} else {
writeToDirectory(output, outputMode, overwrite);
}
}
/**
* Write the dex program resources and proguard resource to @code{directory}.
*/
public void writeToDirectory(Path directory, OutputMode outputMode) throws IOException {
writeToDirectory(directory, outputMode, false);
}
/**
* Write the dex program resources and proguard resource to @code{directory}.
*/
public void writeToDirectory(
Path directory, OutputMode outputMode, boolean overwrite) throws IOException {
CopyOption[] options = copyOptions(overwrite);
try (Closer closer = Closer.create()) {
List<InternalResource> dexProgramSources = getDexProgramResources();
for (int i = 0; i < dexProgramSources.size(); i++) {
Path fileName = directory.resolve(outputMode.getFileName(dexProgramSources.get(i), i));
Files.copy(dexProgramSources.get(i).getStream(closer), fileName, options);
}
writeProguardMap(closer, directory, overwrite);
writeProguardSeeds(closer, directory, overwrite);
}
}
public List<byte[]> writeToMemory() throws IOException {
List<byte[]> dex = new ArrayList<>();
try (Closer closer = Closer.create()) {
List<InternalResource> dexProgramSources = getDexProgramResources();
for (int i = 0; i < dexProgramSources.size(); i++) {
ByteArrayOutputStream out = new ByteArrayOutputStream();
ByteStreams.copy(dexProgramSources.get(i).getStream(closer), out);
dex.add(out.toByteArray());
}
// TODO(sgjesse): Add Proguard map and seeds.
}
return dex;
}
/**
* Write the dex program resources to @code{archive} and the proguard resource as its sibling.
*/
public void writeToZip(Path archive, OutputMode outputMode) throws IOException {
writeToZip(archive, outputMode, false);
}
/**
* Write the dex program resources to @code{archive} and the proguard resource as its sibling.
*/
public void writeToZip(
Path archive, OutputMode outputMode, boolean overwrite) throws IOException {
OpenOption[] options = openOptions(overwrite);
try (Closer closer = Closer.create()) {
try (ZipOutputStream out = new ZipOutputStream(Files.newOutputStream(archive, options))) {
List<InternalResource> dexProgramSources = getDexProgramResources();
for (int i = 0; i < dexProgramSources.size(); i++) {
ZipEntry zipEntry = new ZipEntry(outputMode.getFileName(dexProgramSources.get(i), i));
byte[] bytes = ByteStreams.toByteArray(dexProgramSources.get(i).getStream(closer));
zipEntry.setSize(bytes.length);
out.putNextEntry(zipEntry);
out.write(bytes);
out.closeEntry();
}
}
// Write the proguard map to the archives containing directory.
// TODO(zerny): How do we want to determine the output location for the extra resources?
writeProguardMap(closer, archive.getParent(), overwrite);
writeProguardSeeds(closer, archive.getParent(), overwrite);
}
}
private void writeProguardMap(Closer closer, Path parent, boolean overwrite) throws IOException {
InputStream input = getProguardMap(closer);
if (input != null) {
Files.copy(input, parent.resolve(DEFAULT_PROGUARD_MAP_FILE), copyOptions(overwrite));
}
}
public void writeProguardMap(Closer closer, OutputStream out) throws IOException {
InputStream input = getProguardMap(closer);
assert input != null;
out.write(ByteStreams.toByteArray(input));
}
private void writeProguardSeeds(Closer closer, Path parent, boolean overwrite)
throws IOException {
InputStream input = getProguardSeeds(closer);
if (input != null) {
Files.copy(input, parent.resolve(DEFAULT_PROGUARD_SEEDS_FILE), copyOptions(overwrite));
}
}
private OpenOption[] openOptions(boolean overwrite) {
return new OpenOption[]{
overwrite ? StandardOpenOption.CREATE : StandardOpenOption.CREATE_NEW,
StandardOpenOption.TRUNCATE_EXISTING
};
}
private CopyOption[] copyOptions(boolean overwrite) {
return overwrite
? new CopyOption[]{StandardCopyOption.REPLACE_EXISTING}
: new CopyOption[]{};
}
/**
* Builder interface for constructing an AndroidApp.
*/
public static class Builder {
private final List<InternalResource> dexSources = new ArrayList<>();
private final List<InternalResource> classSources = new ArrayList<>();
private final List<ResourceProvider> resourceProviders = new ArrayList<>();
private InternalResource proguardMap;
private InternalResource proguardSeeds;
private InternalResource packageDistribution;
private InternalResource mainDexList;
// See AndroidApp::builder().
private Builder() {
}
// See AndroidApp::builder(AndroidApp).
private Builder(AndroidApp app) {
dexSources.addAll(app.dexSources);
classSources.addAll(app.classSources);
resourceProviders.addAll(app.resourceProviders);
proguardMap = app.proguardMap;
proguardSeeds = app.proguardSeeds;
packageDistribution = app.packageDistribution;
mainDexList = app.mainDexList;
}
/**
* Add dex program files and proguard-map file located in @code{directory}.
*
* <p>The program files included are the top-level files ending in .dex and the proguard-map
* file should it exist (see @code{DEFAULT_PROGUARD_MAP_FILE} for its assumed name).
*
* <p>This method is mostly a utility for reading in the file content produces by some external
* tool, eg, dx.
*
* @param directory Directory containing dex program files and optional proguard-map file.
*/
public Builder addProgramDirectory(Path directory) throws IOException {
File[] resources = directory.toFile().listFiles(file -> isDexFile(file.toPath()));
for (File source : resources) {
addFile(source.toPath(), Resource.Kind.PROGRAM);
}
File mapFile = new File(directory.toFile(), DEFAULT_PROGUARD_MAP_FILE);
if (mapFile.exists()) {
setProguardMapFile(mapFile.toPath());
}
return this;
}
/**
* Add program file resources.
*/
public Builder addProgramFiles(Path... files) throws IOException {
return addProgramFiles(Arrays.asList(files));
}
/**
* Add program file resources.
*/
public Builder addProgramFiles(Collection<Path> files) throws IOException {
for (Path file : files) {
addFile(file, Resource.Kind.PROGRAM);
}
return this;
}
/**
* Add classpath file resources.
*/
public Builder addClasspathFiles(Path... files) throws IOException {
return addClasspathFiles(Arrays.asList(files));
}
/**
* Add classpath file resources.
*/
public Builder addClasspathFiles(Collection<Path> files) throws IOException {
for (Path file : files) {
addFile(file, Resource.Kind.CLASSPATH);
}
return this;
}
/**
* Add classpath resource provider.
*/
public Builder addClasspathResourceProvider(ResourceProvider provider) {
resourceProviders.add(provider);
return this;
}
/**
* Add library file resources.
*/
public Builder addLibraryFiles(Path... files) throws IOException {
return addLibraryFiles(Arrays.asList(files));
}
/**
* Add library file resources.
*/
public Builder addLibraryFiles(Collection<Path> files) throws IOException {
for (Path file : files) {
addFile(file, Resource.Kind.LIBRARY);
}
return this;
}
/**
* Add library resource provider.
*/
public Builder addLibraryResourceProvider(ResourceProvider provider) {
resourceProviders.add(provider);
return this;
}
/**
* Add dex program-data with class descriptor.
*/
public Builder addDexProgramData(byte[] data, Set<String> classDescriptors) {
dexSources.add(InternalResource.fromBytes(Resource.Kind.PROGRAM, data, classDescriptors));
return this;
}
/**
* Add dex program-data.
*/
public Builder addDexProgramData(byte[]... data) {
return addDexProgramData(Arrays.asList(data));
}
/**
* Add dex program-data.
*/
public Builder addDexProgramData(Collection<byte[]> data) {
for (byte[] datum : data) {
dexSources.add(InternalResource.fromBytes(Resource.Kind.PROGRAM, datum));
}
return this;
}
/**
* Add Java-bytecode program data.
*/
public Builder addClassProgramData(byte[]... data) {
return addClassProgramData(Arrays.asList(data));
}
/**
* Add Java-bytecode program data.
*/
public Builder addClassProgramData(Collection<byte[]> data) {
for (byte[] datum : data) {
classSources.add(InternalResource.fromBytes(Resource.Kind.PROGRAM, datum));
}
return this;
}
/**
* Set proguard-map file.
*/
public Builder setProguardMapFile(Path file) {
proguardMap = file == null ? null : InternalResource.fromFile(null, file);
return this;
}
/**
* Set proguard-map data.
*/
public Builder setProguardMapData(String content) {
return setProguardMapData(content == null ? null : content.getBytes(StandardCharsets.UTF_8));
}
/**
* Set proguard-map data.
*/
public Builder setProguardMapData(byte[] content) {
proguardMap = content == null ? null : InternalResource.fromBytes(null, content);
return this;
}
/**
* Set proguard-seeds data.
*/
public Builder setProguardSeedsData(byte[] content) {
proguardSeeds = content == null ? null : InternalResource.fromBytes(null, content);
return this;
}
/**
* Set the package-distribution file.
*/
public Builder setPackageDistributionFile(Path file) {
packageDistribution = file == null ? null : InternalResource.fromFile(null, file);
return this;
}
/**
* Set the main-dex list file.
*/
public Builder setMainDexListFile(Path file) {
mainDexList = file == null ? null : InternalResource.fromFile(null, file);
return this;
}
/**
* Build final AndroidApp.
*/
public AndroidApp build() {
return new AndroidApp(
ImmutableList.copyOf(dexSources),
ImmutableList.copyOf(classSources),
ImmutableList.copyOf(resourceProviders),
proguardMap,
proguardSeeds,
packageDistribution,
mainDexList);
}
private void addFile(Path file, Resource.Kind kind) throws IOException {
if (!Files.exists(file)) {
throw new FileNotFoundException("Non-existent input file: " + file);
}
if (isDexFile(file)) {
dexSources.add(InternalResource.fromFile(kind, file));
} else if (isClassFile(file)) {
classSources.add(InternalResource.fromFile(kind, file));
} else if (isArchive(file)) {
addArchive(file, kind);
} else {
throw new CompilationError("Unsupported source file type for file: " + file);
}
}
private void addArchive(Path archive, Resource.Kind kind) throws IOException {
assert isArchive(archive);
boolean containsDexData = false;
boolean containsClassData = false;
try (ZipInputStream stream = new ZipInputStream(new FileInputStream(archive.toFile()))) {
ZipEntry entry;
while ((entry = stream.getNextEntry()) != null) {
Path name = Paths.get(entry.getName());
if (isDexFile(name)) {
containsDexData = true;
dexSources.add(InternalResource.fromBytes(kind, ByteStreams.toByteArray(stream)));
} else if (isClassFile(name)) {
containsClassData = true;
String descriptor = PreloadedResourceProvider.guessTypeDescriptor(name);
classSources.add(InternalResource.fromBytes(
kind, ByteStreams.toByteArray(stream), Collections.singleton(descriptor)));
}
}
} catch (ZipException e) {
throw new CompilationError(
"Zip error while reading '" + archive + "': " + e.getMessage(), e);
}
if (containsDexData && containsClassData) {
throw new CompilationError(
"Cannot create android app from an archive '" + archive
+ "' containing both DEX and Java-bytecode content");
}
}
}
}