| // Copyright (c) 2019, 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 dx; |
| |
| import com.google.common.base.Optional; |
| import java.io.File; |
| import java.io.IOException; |
| import java.util.ArrayList; |
| import java.util.List; |
| import java.util.Set; |
| import javax.inject.Inject; |
| import org.gradle.api.Action; |
| import org.gradle.api.DefaultTask; |
| import org.gradle.api.UncheckedIOException; |
| import org.gradle.api.file.FileCollection; |
| import org.gradle.api.tasks.InputFile; |
| import org.gradle.api.tasks.InputFiles; |
| import org.gradle.api.tasks.OutputFile; |
| import org.gradle.api.tasks.TaskAction; |
| import org.gradle.process.ExecSpec; |
| import org.gradle.workers.IsolationMode; |
| import org.gradle.workers.WorkerConfiguration; |
| import org.gradle.workers.WorkerExecutor; |
| import utils.Utils; |
| |
| public class DexMergerTask extends DefaultTask { |
| |
| private final WorkerExecutor workerExecutor; |
| |
| private FileCollection source; |
| private File destination; |
| private Optional<File> dexMergerExecutable = Optional.absent(); // Worker API cannot handle null. |
| |
| @Inject |
| public DexMergerTask(WorkerExecutor workerExecutor) { |
| this.workerExecutor = workerExecutor; |
| } |
| |
| @InputFiles |
| public FileCollection getSource() { |
| return source; |
| } |
| |
| public void setSource(FileCollection source) { |
| this.source = source; |
| } |
| |
| @OutputFile |
| public File getDestination() { |
| return destination; |
| } |
| |
| public void setDestination(File destination) { |
| this.destination = destination; |
| } |
| |
| @InputFile |
| @org.gradle.api.tasks.Optional |
| public File getDexMergerExecutable() { |
| return dexMergerExecutable.orNull(); |
| } |
| |
| public void setDexMergerExecutable(File dexMergerExecutable) { |
| this.dexMergerExecutable = Optional.fromNullable(dexMergerExecutable); |
| } |
| |
| @TaskAction |
| void exec() { |
| workerExecutor.submit(RunDexMerger.class, config -> { |
| config.setIsolationMode(IsolationMode.NONE); |
| config.params(source.getFiles(), destination, dexMergerExecutable); |
| }); |
| } |
| |
| public static class RunDexMerger implements Runnable { |
| private final Set<File> sources; |
| private final File destination; |
| private final Optional<File> dexMergerExecutable; |
| |
| @Inject |
| public RunDexMerger(Set<File> sources, File destination, Optional<File> dexMergerExecutable) { |
| this.sources = sources; |
| this.destination = destination; |
| this.dexMergerExecutable = dexMergerExecutable; |
| } |
| |
| @Override |
| public void run() { |
| try { |
| List<String> command = new ArrayList<>(); |
| command.add(dexMergerExecutable.or(Utils::dexMergerExecutable).getCanonicalPath()); |
| command.add(destination.getCanonicalPath()); |
| for (File source : sources) { |
| command.add(source.getCanonicalPath()); |
| } |
| |
| Process dexMerger = new ProcessBuilder(command).inheritIO().start(); |
| int exitCode = dexMerger.waitFor(); |
| if (exitCode != 0) { |
| throw new RuntimeException("Dex merger failed with code " + exitCode); |
| } |
| } catch (IOException e) { |
| throw new java.io.UncheckedIOException(e); |
| } catch (InterruptedException e) { |
| throw new RuntimeException(e); |
| } |
| } |
| } |
| } |