Allow Folder of class files in classpath

Change-Id: I3a1377a609d9fe3aecc056d081f5199d8946b5f6
diff --git a/src/main/java/com/android/tools/r8/BaseCommand.java b/src/main/java/com/android/tools/r8/BaseCommand.java
index 3141dfe..ec4f527 100644
--- a/src/main/java/com/android/tools/r8/BaseCommand.java
+++ b/src/main/java/com/android/tools/r8/BaseCommand.java
@@ -131,15 +131,9 @@
       return self();
     }
 
-    /** Add classpath file resources. */
-    public B addClasspathFiles(Path... files) throws IOException {
-      app.addClasspathFiles(files);
-      return self();
-    }
-
-    /** Add classpath file resources. */
-    public B addClasspathFiles(Collection<Path> files) throws IOException {
-      app.addClasspathFiles(files);
+    /** Add library file resource provider. */
+    public B addLibraryResourceProvider(ClassFileResourceProvider provider) {
+      getAppBuilder().addLibraryResourceProvider(provider);
       return self();
     }
 
diff --git a/src/main/java/com/android/tools/r8/D8Command.java b/src/main/java/com/android/tools/r8/D8Command.java
index 941a890..cbd16fd 100644
--- a/src/main/java/com/android/tools/r8/D8Command.java
+++ b/src/main/java/com/android/tools/r8/D8Command.java
@@ -3,19 +3,16 @@
 // BSD-style license that can be found in the LICENSE file.
 package com.android.tools.r8;
 
-import static com.android.tools.r8.utils.FileUtils.isArchive;
-
+import com.android.tools.r8.errors.CompilationError;
 import com.android.tools.r8.graph.DexItemFactory;
 import com.android.tools.r8.utils.AndroidApp;
 import com.android.tools.r8.utils.InternalOptions;
 import com.android.tools.r8.utils.OffOrAuto;
 import com.android.tools.r8.utils.OutputMode;
-import com.android.tools.r8.utils.PreloadedClassFileProvider;
 import com.google.common.collect.ImmutableList;
 import java.io.IOException;
 import java.nio.file.Path;
 import java.nio.file.Paths;
-import java.util.Arrays;
 import java.util.Collection;
 
 /**
@@ -46,35 +43,13 @@
 
     /** Add classpath file resources. */
     public Builder addClasspathFiles(Path... files) throws IOException {
-      return addClasspathFiles(Arrays.asList(files));
+      getAppBuilder().addClasspathFiles(files);
+      return this;
     }
 
     /** Add classpath file resources. */
     public Builder addClasspathFiles(Collection<Path> files) throws IOException {
-      for (Path file : files) {
-        if (isArchive(file)) {
-          addClasspathResourceProvider(PreloadedClassFileProvider.fromArchive(file));
-        } else {
-          super.addClasspathFiles(file);
-        }
-      }
-      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) {
-        if (isArchive(file)) {
-          addLibraryResourceProvider(PreloadedClassFileProvider.fromArchive(file));
-        } else {
-          super.addLibraryFiles(file);
-        }
-      }
+      getAppBuilder().addClasspathFiles(files);
       return this;
     }
 
@@ -83,11 +58,6 @@
       return this;
     }
 
-    public Builder addLibraryResourceProvider(ClassFileResourceProvider provider) {
-      getAppBuilder().addLibraryResourceProvider(provider);
-      return this;
-    }
-
     @Override
     Builder self() {
       return this;
@@ -136,49 +106,53 @@
     CompilationMode modeSet = null;
     Path outputPath = null;
     Builder builder = builder();
-    for (int i = 0; i < args.length; i++) {
-      String arg = args[i].trim();
-      if (arg.length() == 0) {
-        continue;
-      } else if (arg.equals("--help")) {
-        builder.setPrintHelp(true);
-      } else if (arg.equals("--version")) {
-        builder.setPrintVersion(true);
-      } else if (arg.equals("--debug")) {
-        if (modeSet == CompilationMode.RELEASE) {
-          throw new CompilationException("Cannot compile in both --debug and --release mode.");
+    try {
+      for (int i = 0; i < args.length; i++) {
+        String arg = args[i].trim();
+        if (arg.length() == 0) {
+          continue;
+        } else if (arg.equals("--help")) {
+          builder.setPrintHelp(true);
+        } else if (arg.equals("--version")) {
+          builder.setPrintVersion(true);
+        } else if (arg.equals("--debug")) {
+          if (modeSet == CompilationMode.RELEASE) {
+            throw new CompilationException("Cannot compile in both --debug and --release mode.");
+          }
+          builder.setMode(CompilationMode.DEBUG);
+          modeSet = CompilationMode.DEBUG;
+        } else if (arg.equals("--release")) {
+          if (modeSet == CompilationMode.DEBUG) {
+            throw new CompilationException("Cannot compile in both --debug and --release mode.");
+          }
+          builder.setMode(CompilationMode.RELEASE);
+          modeSet = CompilationMode.RELEASE;
+        } else if (arg.equals("--file-per-class")) {
+          builder.setOutputMode(OutputMode.FilePerClass);
+        } else if (arg.equals("--output")) {
+          String output = args[++i];
+          if (outputPath != null) {
+            throw new CompilationException(
+                "Cannot output both to '" + outputPath.toString() + "' and '" + output + "'");
+          }
+          outputPath = Paths.get(output);
+        } else if (arg.equals("--lib")) {
+          builder.addLibraryFiles(Paths.get(args[++i]));
+        } else if (arg.equals("--classpath")) {
+          builder.addClasspathFiles(Paths.get(args[++i]));
+        } else if (arg.equals("--min-api")) {
+          builder.setMinApiLevel(Integer.valueOf(args[++i]));
+        } else {
+          if (arg.startsWith("--")) {
+            throw new CompilationException("Unknown option: " + arg);
+          }
+          builder.addProgramFiles(Paths.get(arg));
         }
-        builder.setMode(CompilationMode.DEBUG);
-        modeSet = CompilationMode.DEBUG;
-      } else if (arg.equals("--release")) {
-        if (modeSet == CompilationMode.DEBUG) {
-          throw new CompilationException("Cannot compile in both --debug and --release mode.");
-        }
-        builder.setMode(CompilationMode.RELEASE);
-        modeSet = CompilationMode.RELEASE;
-      } else if (arg.equals("--file-per-class")) {
-        builder.setOutputMode(OutputMode.FilePerClass);
-      } else if (arg.equals("--output")) {
-        String output = args[++i];
-        if (outputPath != null) {
-          throw new CompilationException(
-              "Cannot output both to '" + outputPath.toString() + "' and '" + output + "'");
-        }
-        outputPath = Paths.get(output);
-      } else if (arg.equals("--lib")) {
-        builder.addLibraryFiles(Paths.get(args[++i]));
-      } else if (arg.equals("--classpath")) {
-        builder.addClasspathFiles(Paths.get(args[++i]));
-      } else if (arg.equals("--min-api")) {
-        builder.setMinApiLevel(Integer.valueOf(args[++i]));
-      } else {
-        if (arg.startsWith("--")) {
-          throw new CompilationException("Unknown option: " + arg);
-        }
-        builder.addProgramFiles(Paths.get(arg));
       }
+      return builder.setOutputPath(outputPath);
+    } catch (CompilationError e) {
+      throw new CompilationException(e.getMessage(), e);
     }
-    return builder.setOutputPath(outputPath);
   }
 
   private D8Command(
diff --git a/src/main/java/com/android/tools/r8/dex/ApplicationReader.java b/src/main/java/com/android/tools/r8/dex/ApplicationReader.java
index c7a66c2..36b65d7 100644
--- a/src/main/java/com/android/tools/r8/dex/ApplicationReader.java
+++ b/src/main/java/com/android/tools/r8/dex/ApplicationReader.java
@@ -213,11 +213,7 @@
 
     void readSources() throws IOException, ExecutionException {
       readDexSources(inputApp.getDexProgramResources(), PROGRAM, programClasses);
-      readDexSources(inputApp.getDexClasspathResources(), CLASSPATH, classpathClasses);
-      readDexSources(inputApp.getDexLibraryResources(), LIBRARY, libraryClasses);
       readClassSources(inputApp.getClassProgramResources(), PROGRAM, programClasses);
-      readClassSources(inputApp.getClassClasspathResources(), CLASSPATH, classpathClasses);
-      readClassSources(inputApp.getClassLibraryResources(), LIBRARY, libraryClasses);
     }
 
     private <T extends DexClass> ClassProvider<T> buildClassProvider(ClassKind classKind,
diff --git a/src/main/java/com/android/tools/r8/graph/DexApplication.java b/src/main/java/com/android/tools/r8/graph/DexApplication.java
index e57cd05..2c9366c 100644
--- a/src/main/java/com/android/tools/r8/graph/DexApplication.java
+++ b/src/main/java/com/android/tools/r8/graph/DexApplication.java
@@ -126,6 +126,9 @@
   }
 
   public DexClass definitionFor(DexType type) {
+    if (type == null) {
+      return null;
+    }
     DexClass clazz = programClasses.get(type);
     if (clazz == null && classpathClasses != null) {
       clazz = classpathClasses.get(type);
diff --git a/src/main/java/com/android/tools/r8/utils/AndroidApp.java b/src/main/java/com/android/tools/r8/utils/AndroidApp.java
index 7d8bada..3877f1a 100644
--- a/src/main/java/com/android/tools/r8/utils/AndroidApp.java
+++ b/src/main/java/com/android/tools/r8/utils/AndroidApp.java
@@ -10,8 +10,6 @@
 import com.android.tools.r8.ClassFileResourceProvider;
 import com.android.tools.r8.Resource;
 import com.android.tools.r8.errors.CompilationError;
-import com.android.tools.r8.errors.Unreachable;
-import com.android.tools.r8.graph.ClassKind;
 import com.google.common.collect.ImmutableList;
 import com.google.common.io.ByteStreams;
 import com.google.common.io.Closer;
@@ -52,8 +50,6 @@
   public static final String DEFAULT_PROGUARD_MAP_FILE = "proguard.map";
 
   private final ImmutableList<Resource> programResources;
-  private final ImmutableList<Resource> classpathResources;
-  private final ImmutableList<Resource> libraryResources;
   private final ImmutableList<ClassFileResourceProvider> classpathResourceProviders;
   private final ImmutableList<ClassFileResourceProvider> libraryResourceProviders;
   private final Resource proguardMap;
@@ -64,8 +60,6 @@
   // See factory methods and AndroidApp.Builder below.
   private AndroidApp(
       ImmutableList<Resource> programResources,
-      ImmutableList<Resource> classpathResources,
-      ImmutableList<Resource> libraryResources,
       ImmutableList<ClassFileResourceProvider> classpathResourceProviders,
       ImmutableList<ClassFileResourceProvider> libraryResourceProviders,
       Resource proguardMap,
@@ -73,8 +67,6 @@
       Resource packageDistribution,
       Resource mainDexList) {
     this.programResources = programResources;
-    this.classpathResources = classpathResources;
-    this.libraryResources = libraryResources;
     this.classpathResourceProviders = classpathResourceProviders;
     this.libraryResourceProviders = libraryResourceProviders;
     this.proguardMap = proguardMap;
@@ -156,26 +148,6 @@
     return filter(programResources, Resource.Kind.CLASSFILE);
   }
 
-  /** Get input streams for all dex program classpath resources. */
-  public List<Resource> getDexClasspathResources() {
-    return filter(classpathResources, Resource.Kind.DEX);
-  }
-
-  /** Get input streams for all Java-bytecode classpath resources. */
-  public List<Resource> getClassClasspathResources() {
-    return filter(classpathResources, Resource.Kind.CLASSFILE);
-  }
-
-  /** Get input streams for all dex library resources. */
-  public List<Resource> getDexLibraryResources() {
-    return filter(libraryResources, Resource.Kind.DEX);
-  }
-
-  /** Get input streams for all Java-bytecode library resources. */
-  public List<Resource> getClassLibraryResources() {
-    return filter(libraryResources, Resource.Kind.CLASSFILE);
-  }
-
   /** Get classpath resource providers. */
   public List<ClassFileResourceProvider> getClasspathResourceProviders() {
     return classpathResourceProviders;
@@ -371,8 +343,6 @@
   public static class Builder {
 
     private final List<Resource> programResources = new ArrayList<>();
-    private final List<Resource> classpathResources = new ArrayList<>();
-    private final List<Resource> libraryResources = new ArrayList<>();
     private final List<ClassFileResourceProvider> classpathResourceProviders = new ArrayList<>();
     private final List<ClassFileResourceProvider> libraryResourceProviders = new ArrayList<>();
     private Resource proguardMap;
@@ -387,8 +357,6 @@
     // See AndroidApp::builder(AndroidApp).
     private Builder(AndroidApp app) {
       programResources.addAll(app.programResources);
-      classpathResources.addAll(app.classpathResources);
-      libraryResources.addAll(app.libraryResources);
       classpathResourceProviders.addAll(app.classpathResourceProviders);
       libraryResourceProviders.addAll(app.libraryResourceProviders);
       proguardMap = app.proguardMap;
@@ -409,10 +377,10 @@
      * @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(), ClassKind.PROGRAM);
-      }
+      List<Path> resources =
+          Arrays.asList(directory.toFile().listFiles(file -> isDexFile(file.toPath()))).stream()
+          .map(file -> file.toPath()).collect(Collectors.toList());
+      addProgramFiles(resources);
       File mapFile = new File(directory.toFile(), DEFAULT_PROGUARD_MAP_FILE);
       if (mapFile.exists()) {
         setProguardMapFile(mapFile.toPath());
@@ -432,7 +400,7 @@
      */
     public Builder addProgramFiles(Collection<Path> files) throws IOException {
       for (Path file : files) {
-        addFile(file, ClassKind.PROGRAM);
+        addProgramFile(file);
       }
       return this;
     }
@@ -449,7 +417,7 @@
      */
     public Builder addClasspathFiles(Collection<Path> files) throws IOException {
       for (Path file : files) {
-        addFile(file, ClassKind.CLASSPATH);
+        addClassProvider(file, classpathResourceProviders);
       }
       return this;
     }
@@ -474,7 +442,7 @@
      */
     public Builder addLibraryFiles(Collection<Path> files) throws IOException {
       for (Path file : files) {
-        addFile(file, ClassKind.LIBRARY);
+        addClassProvider(file, libraryResourceProviders);
       }
       return this;
     }
@@ -491,7 +459,7 @@
      * Add dex program-data with class descriptor.
      */
     public Builder addDexProgramData(byte[] data, Set<String> classDescriptors) {
-      resources(ClassKind.PROGRAM).add(
+      programResources.add(
           Resource.fromBytes(Resource.Kind.DEX, data, classDescriptors));
       return this;
     }
@@ -508,7 +476,7 @@
      */
     public Builder addDexProgramData(Collection<byte[]> data) {
       for (byte[] datum : data) {
-        resources(ClassKind.PROGRAM).add(Resource.fromBytes(Resource.Kind.DEX, datum));
+        programResources.add(Resource.fromBytes(Resource.Kind.DEX, datum));
       }
       return this;
     }
@@ -525,7 +493,7 @@
      */
     public Builder addClassProgramData(Collection<byte[]> data) {
       for (byte[] datum : data) {
-        resources(ClassKind.PROGRAM).add(Resource.fromBytes(Resource.Kind.CLASSFILE, datum));
+        programResources.add(Resource.fromBytes(Resource.Kind.CLASSFILE, datum));
       }
       return this;
     }
@@ -591,8 +559,6 @@
     public AndroidApp build() {
       return new AndroidApp(
           ImmutableList.copyOf(programResources),
-          ImmutableList.copyOf(classpathResources),
-          ImmutableList.copyOf(libraryResources),
           ImmutableList.copyOf(classpathResourceProviders),
           ImmutableList.copyOf(libraryResourceProviders),
           proguardMap,
@@ -601,34 +567,36 @@
           mainDexList);
     }
 
-    private List<Resource> resources(ClassKind classKind) {
-      switch (classKind) {
-        case PROGRAM:
-          return programResources;
-        case CLASSPATH:
-          return classpathResources;
-        case LIBRARY:
-          return libraryResources;
-      }
-      throw new Unreachable();
-    }
-
-    private void addFile(Path file, ClassKind classKind) throws IOException {
+    private void addProgramFile(Path file) throws IOException {
       if (!Files.exists(file)) {
         throw new FileNotFoundException("Non-existent input file: " + file);
       }
       if (isDexFile(file)) {
-        resources(classKind).add(Resource.fromFile(Resource.Kind.DEX, file));
+        programResources.add(Resource.fromFile(Resource.Kind.DEX, file));
       } else if (isClassFile(file)) {
-        resources(classKind).add(Resource.fromFile(Resource.Kind.CLASSFILE, file));
+        programResources.add(Resource.fromFile(Resource.Kind.CLASSFILE, file));
       } else if (isArchive(file)) {
-        addArchive(file, classKind);
+        addProgramArchive(file);
       } else {
         throw new CompilationError("Unsupported source file type for file: " + file);
       }
     }
 
-    private void addArchive(Path archive, ClassKind classKind) throws IOException {
+    private void addClassProvider(Path file, List<ClassFileResourceProvider> providerList)
+        throws IOException {
+      if (!Files.exists(file)) {
+        throw new FileNotFoundException("Non-existent input file: " + file);
+      }
+      if (isArchive(file)) {
+        providerList.add(PreloadedClassFileProvider.fromArchive(file));
+      } else if (Files.isDirectory(file) ) {
+        providerList.add(DirectoryClassFileProvider.fromDirectory(file));
+      } else {
+        throw new CompilationError("Unsupported source file type for file: " + file);
+      }
+    }
+
+    private void addProgramArchive(Path archive) throws IOException {
       assert isArchive(archive);
       boolean containsDexData = false;
       boolean containsClassData = false;
@@ -638,12 +606,12 @@
           Path name = Paths.get(entry.getName());
           if (isDexFile(name)) {
             containsDexData = true;
-            resources(classKind).add(Resource.fromBytes(
+            programResources.add(Resource.fromBytes(
                 Resource.Kind.DEX, ByteStreams.toByteArray(stream)));
           } else if (isClassFile(name)) {
             containsClassData = true;
             String descriptor = PreloadedClassFileProvider.guessTypeDescriptor(name);
-            resources(classKind).add(Resource.fromBytes(Resource.Kind.CLASSFILE,
+            programResources.add(Resource.fromBytes(Resource.Kind.CLASSFILE,
                 ByteStreams.toByteArray(stream), Collections.singleton(descriptor)));
           }
         }
diff --git a/src/main/java/com/android/tools/r8/utils/DirectoryClassFileProvider.java b/src/main/java/com/android/tools/r8/utils/DirectoryClassFileProvider.java
index c060ead..7de53d2 100644
--- a/src/main/java/com/android/tools/r8/utils/DirectoryClassFileProvider.java
+++ b/src/main/java/com/android/tools/r8/utils/DirectoryClassFileProvider.java
@@ -69,6 +69,10 @@
     return new DirectoryClassFileProvider(dir.toAbsolutePath());
   }
 
+  public Path getRoot() {
+    return root;
+  }
+
   @Override
   public String toString() {
     return "directory(" + root + ")";
diff --git a/src/main/java/com/android/tools/r8/utils/ZipUtils.java b/src/main/java/com/android/tools/r8/utils/ZipUtils.java
index 59f0702..47d8e01 100644
--- a/src/main/java/com/android/tools/r8/utils/ZipUtils.java
+++ b/src/main/java/com/android/tools/r8/utils/ZipUtils.java
@@ -3,11 +3,13 @@
 // BSD-style license that can be found in the LICENSE file.
 package com.android.tools.r8.utils;
 
+import com.android.tools.r8.errors.CompilationError;
 import com.google.common.io.ByteStreams;
 import java.io.File;
 import java.io.FileInputStream;
 import java.io.FileOutputStream;
 import java.io.IOException;
+import java.nio.file.Path;
 import java.util.ArrayList;
 import java.util.List;
 import java.util.function.Predicate;
@@ -35,11 +37,18 @@
 
   public static List<File> unzip(String zipFile, File outDirectory, Predicate<ZipEntry> filter)
       throws IOException {
+    final Path outDirectoryPath = outDirectory.toPath();
     final List<File> outFiles = new ArrayList<>();
       iter(zipFile, (entry, input) -> {
         String name = entry.getName();
-        if (filter.test(entry)) {
-          File outFile = outDirectory.toPath().resolve(name).toFile();
+        if (!entry.isDirectory() && filter.test(entry)) {
+          if (name.contains("..")) {
+            // Protect against malicious archives.
+            throw new CompilationError("Invalid entry name \"" + name + "\"");
+          }
+          Path outPath = outDirectoryPath.resolve(name);
+          File outFile = outPath.toFile();
+          outFile.getParentFile().mkdirs();
           FileOutputStream output = new FileOutputStream(outFile);
           ByteStreams.copy(input, output);
           outFiles.add(outFile);
diff --git a/src/test/java/com/android/tools/r8/D8LazyRunExamplesAndroidOTest.java b/src/test/java/com/android/tools/r8/D8LazyRunExamplesAndroidOTest.java
index 34cd83c..9c80561 100644
--- a/src/test/java/com/android/tools/r8/D8LazyRunExamplesAndroidOTest.java
+++ b/src/test/java/com/android/tools/r8/D8LazyRunExamplesAndroidOTest.java
@@ -4,10 +4,26 @@
 
 package com.android.tools.r8;
 
+import static com.android.tools.r8.dex.Constants.ANDROID_K_API;
+import static com.android.tools.r8.utils.FileUtils.JAR_EXTENSION;
+import static org.junit.Assert.assertTrue;
+
+import com.android.tools.r8.utils.AndroidApp;
 import com.android.tools.r8.utils.DirectoryClassFileProvider;
+import com.android.tools.r8.utils.FileUtils;
+import com.android.tools.r8.utils.OffOrAuto;
 import com.android.tools.r8.utils.PreloadedClassFileProvider;
+import com.android.tools.r8.utils.ZipUtils;
 import java.io.IOException;
+import java.nio.file.Files;
 import java.nio.file.Path;
+import java.nio.file.Paths;
+import java.util.ArrayList;
+import java.util.Arrays;
+import java.util.Iterator;
+import java.util.List;
+import java.util.stream.Collectors;
+import org.junit.Test;
 
 public class D8LazyRunExamplesAndroidOTest
     extends D8IncrementalRunExamplesAndroidOTest {
@@ -39,4 +55,64 @@
   D8IncrementalTestRunner test(String testName, String packageName, String mainClass) {
     return new D8LazyTestRunner(testName, packageName, mainClass);
   }
+
+  @Test
+  public void dexPerClassFileWithDesugaringAndFolderClasspath() throws Throwable {
+    int minAPILevel = ANDROID_K_API;
+    Path inputFile =
+        Paths.get(ToolHelper.EXAMPLES_ANDROID_N_BUILD_DIR, "interfacemethods" + JAR_EXTENSION);
+    Path tmpClassesDir = temp.newFolder().toPath();
+    ZipUtils.unzip(inputFile.toString(), tmpClassesDir.toFile());
+    Path androidJar = Paths.get(ToolHelper.getAndroidJar(minAPILevel));
+
+    // Build all at once.
+    AndroidApp fullBuildResult;
+    {
+      D8Command command = D8Command.builder()
+          .setMinApiLevel(minAPILevel)
+          .addLibraryFiles(androidJar)
+          .addProgramFiles(inputFile)
+          .build();
+
+      fullBuildResult = ToolHelper.runD8(
+          command, (options) -> options.interfaceMethodDesugaring = OffOrAuto.Auto);
+    }
+
+    // Build each class individually using tmpClassesDir as classpath for desugaring.
+    List<Resource> individalDexes = new ArrayList<>();
+    List<Path> individualClassFiles =
+        Files.walk(tmpClassesDir)
+        .filter(classFile -> FileUtils.isClassFile(classFile))
+        .collect(Collectors.toList());
+    for (Path classFile : individualClassFiles) {
+      D8Command.Builder builder =
+          D8Command.builder()
+              .setMinApiLevel(minAPILevel)
+              .addLibraryFiles(androidJar)
+              .addClasspathFiles(tmpClassesDir)
+              .addProgramFiles(classFile);
+      AndroidApp individualResult =
+          ToolHelper.runD8(
+              builder.build(),
+              (options) -> options.interfaceMethodDesugaring = OffOrAuto.Auto);
+      individalDexes.add(individualResult.getDexProgramResources().get(0));
+    }
+    AndroidApp mergedResult = mergeDexResources(minAPILevel, individalDexes);
+
+    assertTrue(Arrays.equals(
+        readFromResource(fullBuildResult.getDexProgramResources().get(0)),
+        readFromResource(mergedResult.getDexProgramResources().get(0))));
+  }
+
+  private AndroidApp mergeDexResources(int minAPILevel, List<Resource> individalDexes)
+      throws IOException, CompilationException {
+    D8Command.Builder builder = D8Command.builder()
+        .setMinApiLevel(minAPILevel);
+    for (Resource resource : individalDexes) {
+      builder.addDexProgramData(readFromResource(resource));
+    }
+    AndroidApp mergedResult = ToolHelper.runD8(builder.build());
+    return mergedResult;
+  }
+
 }
diff --git a/src/test/java/com/android/tools/r8/ToolHelper.java b/src/test/java/com/android/tools/r8/ToolHelper.java
index e74eb2c..e01ecc0 100644
--- a/src/test/java/com/android/tools/r8/ToolHelper.java
+++ b/src/test/java/com/android/tools/r8/ToolHelper.java
@@ -455,7 +455,7 @@
         throws ProguardRuleParserException, ExecutionException, IOException {
    // TODO(zerny): Should we really be adding the android library in ToolHelper?
     AndroidApp app = command.getInputApp();
-    if (app.getClassLibraryResources().isEmpty()) {
+    if (app.getLibraryResourceProviders().isEmpty()) {
       app =
           AndroidApp.builder(app)
               .addLibraryFiles(Paths.get(getAndroidJar(command.getMinApiLevel())))
diff --git a/src/test/java/com/android/tools/r8/memberrebinding/MemberRebindingTest.java b/src/test/java/com/android/tools/r8/memberrebinding/MemberRebindingTest.java
index c617c25..a80f46d 100644
--- a/src/test/java/com/android/tools/r8/memberrebinding/MemberRebindingTest.java
+++ b/src/test/java/com/android/tools/r8/memberrebinding/MemberRebindingTest.java
@@ -40,10 +40,8 @@
 public class MemberRebindingTest {
 
   private static final String ANDROID_JAR = ToolHelper.getDefaultAndroidJar();
-  private static final List<String> JAR_LIBRARIES = ImmutableList
-      .of(ANDROID_JAR, ToolHelper.EXAMPLES_BUILD_DIR + "memberrebindinglib.jar");
-  private static final List<String> DEX_LIBRARIES = ImmutableList
-      .of(ANDROID_JAR, ToolHelper.EXAMPLES_BUILD_DIR + "memberrebindinglib/classes.dex");
+  private static final List<Path> JAR_LIBRARIES = ListUtils.map(ImmutableList
+      .of(ANDROID_JAR, ToolHelper.EXAMPLES_BUILD_DIR + "memberrebindinglib.jar"), Paths::get);
 
   private enum Frontend {
     DEX, JAR;
@@ -82,14 +80,13 @@
       throws IOException, ProguardRuleParserException, ExecutionException, CompilationException {
     // Generate R8 processed version without library option.
     String out = temp.getRoot().getCanonicalPath();
-    List<String> libs = kind == Frontend.DEX ? DEX_LIBRARIES : JAR_LIBRARIES;
     // NOTE: It is important to turn off inlining to ensure
     // dex inspection of invokes is predictable.
     ToolHelper.runR8(
         R8Command.builder()
             .setOutputPath(Paths.get(out))
             .addProgramFiles(programFile)
-            .addLibraryFiles(ListUtils.map(libs, Paths::get))
+            .addLibraryFiles(JAR_LIBRARIES)
             .setMinApiLevel(minApiLevel)
             .build(),
         options -> options.inlineAccessors = false);
diff --git a/src/test/java/com/android/tools/r8/shaking/TreeShakingTest.java b/src/test/java/com/android/tools/r8/shaking/TreeShakingTest.java
index 0904569..3487694 100644
--- a/src/test/java/com/android/tools/r8/shaking/TreeShakingTest.java
+++ b/src/test/java/com/android/tools/r8/shaking/TreeShakingTest.java
@@ -57,10 +57,8 @@
 public class TreeShakingTest {
 
   private static final String ANDROID_JAR = ToolHelper.getDefaultAndroidJar();
-  private static final List<String> JAR_LIBRARIES = ImmutableList
-      .of(ANDROID_JAR, ToolHelper.EXAMPLES_BUILD_DIR + "shakinglib.jar");
-  private static final List<String> DEX_LIBRARIES = ImmutableList
-      .of(ANDROID_JAR, ToolHelper.EXAMPLES_BUILD_DIR + "shakinglib/classes.dex");
+  private static final List<Path> JAR_LIBRARIES = ListUtils.map(ImmutableList
+      .of(ANDROID_JAR, ToolHelper.EXAMPLES_BUILD_DIR + "shakinglib.jar"), Paths::get);
   private static final String EMPTY_FLAGS = "src/test/proguard/valid/empty.flags";
   private static Set<String> IGNORED = ImmutableSet.of(
       // there's no point in running those without obfuscation
@@ -116,7 +114,6 @@
       throws IOException, ProguardRuleParserException, ExecutionException, CompilationException {
     // Generate R8 processed version without library option.
     Path out = temp.getRoot().toPath();
-    List<String> libs = kind == Frontend.DEX ? DEX_LIBRARIES : JAR_LIBRARIES;
     boolean inline = programFile.contains("inlining");
 
     R8Command command =
@@ -124,7 +121,7 @@
             .setOutputPath(out)
             .addProgramFiles(Paths.get(programFile))
             .addProguardConfigurationFiles(ListUtils.map(keepRulesFiles, Paths::get))
-            .addLibraryFiles(ListUtils.map(libs, Paths::get))
+            .addLibraryFiles(JAR_LIBRARIES)
             .setMinification(minify)
             .build();
     ToolHelper.runR8(command, options -> {
diff --git a/src/test/java/com/android/tools/r8/utils/D8CommandTest.java b/src/test/java/com/android/tools/r8/utils/D8CommandTest.java
index 1db48c4..5bae2ed 100644
--- a/src/test/java/com/android/tools/r8/utils/D8CommandTest.java
+++ b/src/test/java/com/android/tools/r8/utils/D8CommandTest.java
@@ -4,6 +4,7 @@
 package com.android.tools.r8.utils;
 
 import static com.android.tools.r8.ToolHelper.EXAMPLES_BUILD_DIR;
+import static com.android.tools.r8.utils.FileUtils.JAR_EXTENSION;
 import static org.junit.Assert.assertEquals;
 import static org.junit.Assert.assertFalse;
 import static org.junit.Assert.assertNull;
@@ -48,8 +49,6 @@
   private void verifyEmptyCommand(D8Command command) {
     assertEquals(0, ToolHelper.getApp(command).getDexProgramResources().size());
     assertEquals(0, ToolHelper.getApp(command).getClassProgramResources().size());
-    assertEquals(0, ToolHelper.getApp(command).getDexLibraryResources().size());
-    assertEquals(0, ToolHelper.getApp(command).getClassLibraryResources().size());
     assertFalse(ToolHelper.getApp(command).hasMainDexList());
     assertFalse(ToolHelper.getApp(command).hasProguardMap());
     assertFalse(ToolHelper.getApp(command).hasProguardSeeds());
@@ -164,6 +163,40 @@
   }
 
   @Test
+  public void folderLibAndClasspath() throws Throwable {
+    Path inputFile =
+        Paths.get(ToolHelper.EXAMPLES_ANDROID_N_BUILD_DIR, "interfacemethods" + JAR_EXTENSION);
+    Path tmpClassesDir = temp.newFolder().toPath();
+    ZipUtils.unzip(inputFile.toString(), tmpClassesDir.toFile());
+    D8Command command = parse("--lib", tmpClassesDir.toString(), "--classpath",
+        tmpClassesDir.toString());
+    AndroidApp inputApp = ToolHelper.getApp(command);
+    assertEquals(1, inputApp.getClasspathResourceProviders().size());
+    assertEquals(tmpClassesDir,
+        ((DirectoryClassFileProvider) inputApp.getClasspathResourceProviders().get(0)).getRoot());
+    assertEquals(1, inputApp.getLibraryResourceProviders().size());
+    assertEquals(tmpClassesDir,
+        ((DirectoryClassFileProvider) inputApp.getLibraryResourceProviders().get(0)).getRoot());
+  }
+
+  @Test
+  public void classFolderProgram() throws Throwable {
+    thrown.expect(CompilationException.class);
+    Path inputFile =
+        Paths.get(ToolHelper.EXAMPLES_ANDROID_N_BUILD_DIR, "interfacemethods" + JAR_EXTENSION);
+    Path tmpClassesDir = temp.newFolder().toPath();
+    ZipUtils.unzip(inputFile.toString(), tmpClassesDir.toFile());
+    parse(tmpClassesDir.toString());
+  }
+
+  @Test
+  public void emptyFolderProgram() throws Throwable {
+    thrown.expect(CompilationException.class);
+    Path tmpClassesDir = temp.newFolder().toPath();
+    parse(tmpClassesDir.toString());
+  }
+
+  @Test
   public void nonExistingOutputJar() throws Throwable {
     Path nonExistingJar = temp.getRoot().toPath().resolve("non-existing-archive.jar");
     D8Command.builder().setOutputPath(nonExistingJar).build();
diff --git a/src/test/java/com/android/tools/r8/utils/R8CommandTest.java b/src/test/java/com/android/tools/r8/utils/R8CommandTest.java
index 2af79cb..0c737b9 100644
--- a/src/test/java/com/android/tools/r8/utils/R8CommandTest.java
+++ b/src/test/java/com/android/tools/r8/utils/R8CommandTest.java
@@ -49,8 +49,6 @@
   private void verifyEmptyCommand(R8Command command) {
     assertEquals(0, ToolHelper.getApp(command).getDexProgramResources().size());
     assertEquals(0, ToolHelper.getApp(command).getClassProgramResources().size());
-    assertEquals(0, ToolHelper.getApp(command).getDexLibraryResources().size());
-    assertEquals(0, ToolHelper.getApp(command).getClassLibraryResources().size());
     assertFalse(ToolHelper.getApp(command).hasMainDexList());
     assertFalse(ToolHelper.getApp(command).hasProguardMap());
     assertFalse(ToolHelper.getApp(command).hasProguardSeeds());