| // Copyright 2015 Google Inc. All rights reserved. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| package java |
| |
| // This file generates the final rules for compiling all Java. All properties related to |
| // compiling should have been translated into javaBuilderFlags or another argument to the Transform* |
| // functions. |
| |
| import ( |
| "path/filepath" |
| "strconv" |
| "strings" |
| |
| "github.com/google/blueprint" |
| "github.com/google/blueprint/proptools" |
| |
| "android/soong/android" |
| "android/soong/remoteexec" |
| ) |
| |
| var ( |
| pctx = android.NewPackageContext("android/soong/java") |
| |
| // Compiling java is not conducive to proper dependency tracking. The path-matches-class-name |
| // requirement leads to unpredictable generated source file names, and a single .java file |
| // will get compiled into multiple .class files if it contains inner classes. To work around |
| // this, all java rules write into separate directories and then are combined into a .jar file |
| // (if the rule produces .class files) or a .srcjar file (if the rule produces .java files). |
| // .srcjar files are unzipped into a temporary directory when compiled with javac. |
| // TODO(b/143658984): goma can't handle the --system argument to javac. |
| javac, javacRE = pctx.MultiCommandRemoteStaticRules("javac", |
| blueprint.RuleParams{ |
| Command: `rm -rf "$outDir" "$annoDir" "$annoSrcJar.tmp" "$srcJarDir" "$out.tmp" && ` + |
| `mkdir -p "$outDir" "$annoDir" "$srcJarDir" && ` + |
| `${config.ZipSyncCmd} -d $srcJarDir -l $srcJarDir/list -f "*.java" $srcJars && ` + |
| `(if [ -s $srcJarDir/list ] || [ -s $out.rsp ] ; then ` + |
| `${config.SoongJavacWrapper} $javaTemplate${config.JavacCmd} ` + |
| `${config.JavacHeapFlags} ${config.JavacVmFlags} ${config.CommonJdkFlags} ` + |
| `$processorpath $processor $javacFlags $bootClasspath $classpath ` + |
| `-source $javaVersion -target $javaVersion ` + |
| `-d $outDir -s $annoDir @$out.rsp @$srcJarDir/list ; fi ) && ` + |
| `$annoSrcJarTemplate${config.SoongZipCmd} -jar -o $annoSrcJar.tmp -C $annoDir -D $annoDir && ` + |
| `$zipTemplate${config.SoongZipCmd} -jar -o $out.tmp -C $outDir -D $outDir && ` + |
| `if ! cmp -s "$out.tmp" "$out"; then mv "$out.tmp" "$out"; fi && ` + |
| `if ! cmp -s "$annoSrcJar.tmp" "$annoSrcJar"; then mv "$annoSrcJar.tmp" "$annoSrcJar"; fi && ` + |
| `rm -rf "$srcJarDir" "$outDir"`, |
| CommandDeps: []string{ |
| "${config.JavacCmd}", |
| "${config.SoongZipCmd}", |
| "${config.ZipSyncCmd}", |
| }, |
| CommandOrderOnly: []string{"${config.SoongJavacWrapper}"}, |
| Restat: true, |
| Rspfile: "$out.rsp", |
| RspfileContent: "$in", |
| }, map[string]*remoteexec.REParams{ |
| "$javaTemplate": &remoteexec.REParams{ |
| Labels: map[string]string{"type": "compile", "lang": "java", "compiler": "javac"}, |
| ExecStrategy: "${config.REJavacExecStrategy}", |
| Platform: map[string]string{remoteexec.PoolKey: "${config.REJavaPool}"}, |
| }, |
| "$zipTemplate": &remoteexec.REParams{ |
| Labels: map[string]string{"type": "tool", "name": "soong_zip"}, |
| Inputs: []string{"${config.SoongZipCmd}", "$outDir"}, |
| OutputFiles: []string{"$out.tmp"}, |
| ExecStrategy: "${config.REJavacExecStrategy}", |
| Platform: map[string]string{remoteexec.PoolKey: "${config.REJavaPool}"}, |
| }, |
| "$annoSrcJarTemplate": &remoteexec.REParams{ |
| Labels: map[string]string{"type": "tool", "name": "soong_zip"}, |
| Inputs: []string{"${config.SoongZipCmd}", "$annoDir"}, |
| OutputFiles: []string{"$annoSrcJar.tmp"}, |
| ExecStrategy: "${config.REJavacExecStrategy}", |
| Platform: map[string]string{remoteexec.PoolKey: "${config.REJavaPool}"}, |
| }, |
| }, []string{"javacFlags", "bootClasspath", "classpath", "processorpath", "processor", "srcJars", "srcJarDir", |
| "outDir", "annoDir", "annoSrcJar", "javaVersion"}, nil) |
| |
| _ = pctx.VariableFunc("kytheCorpus", |
| func(ctx android.PackageVarContext) string { return ctx.Config().XrefCorpusName() }) |
| _ = pctx.VariableFunc("kytheCuEncoding", |
| func(ctx android.PackageVarContext) string { return ctx.Config().XrefCuEncoding() }) |
| _ = pctx.VariableFunc("kytheCuJavaSourceMax", |
| func(ctx android.PackageVarContext) string { return ctx.Config().XrefCuJavaSourceMax() }) |
| _ = pctx.SourcePathVariable("kytheVnames", "build/soong/vnames.json") |
| // Run it with several --add-exports to allow the classes in the |
| // com.google.devtools.kythe.extractors.java.standalone package access the packages in the |
| // jdk.compiler compiler module. Long live Java modules. |
| kytheExtract = pctx.AndroidStaticRule("kythe", |
| blueprint.RuleParams{ |
| Command: `${config.ZipSyncCmd} -d $srcJarDir ` + |
| `-l $srcJarDir/list -f "*.java" $srcJars && ` + |
| `( [ ! -s $srcJarDir/list -a ! -s $out.rsp ] || ` + |
| `KYTHE_ROOT_DIRECTORY=. KYTHE_OUTPUT_FILE=$out ` + |
| `KYTHE_CORPUS=${kytheCorpus} ` + |
| `KYTHE_VNAMES=${kytheVnames} ` + |
| `KYTHE_KZIP_ENCODING=${kytheCuEncoding} ` + |
| `KYTHE_JAVA_SOURCE_BATCH_SIZE=${kytheCuJavaSourceMax} ` + |
| `${config.SoongJavacWrapper} ${config.JavaCmd} ` + |
| // Avoid JDK9's warning about "Illegal reflective access by com.google.protobuf.Utf8$UnsafeProcessor ... |
| // to field java.nio.Buffer.address" |
| `--add-opens=java.base/java.nio=ALL-UNNAMED ` + |
| // Allow the classes in the com.google.devtools.kythe.extractors.java.standalone package |
| // access the packages in the jdk.compiler compiler module |
| `--add-opens=java.base/java.nio=ALL-UNNAMED ` + |
| `--add-exports=jdk.compiler/com.sun.tools.javac.util=ALL-UNNAMED ` + |
| `--add-exports=jdk.compiler/com.sun.tools.javac.main=ALL-UNNAMED ` + |
| `--add-exports=jdk.compiler/com.sun.tools.javac.file=ALL-UNNAMED ` + |
| `--add-exports=jdk.compiler/com.sun.tools.javac.api=ALL-UNNAMED ` + |
| `--add-exports=jdk.compiler/com.sun.tools.javac.code=ALL-UNNAMED ` + |
| `-jar ${config.JavaKytheExtractorJar} ` + |
| `${config.JavacHeapFlags} ${config.CommonJdkFlags} ` + |
| `$processorpath $processor $javacFlags $bootClasspath $classpath ` + |
| `-source $javaVersion -target $javaVersion ` + |
| `-d $outDir -s $annoDir @$out.rsp @$srcJarDir/list)`, |
| CommandDeps: []string{ |
| "${config.JavaCmd}", |
| "${config.JavaKytheExtractorJar}", |
| "${kytheVnames}", |
| "${config.ZipSyncCmd}", |
| }, |
| CommandOrderOnly: []string{"${config.SoongJavacWrapper}"}, |
| Rspfile: "$out.rsp", |
| RspfileContent: "$in", |
| }, |
| "javacFlags", "bootClasspath", "classpath", "processorpath", "processor", "srcJars", "srcJarDir", |
| "outDir", "annoDir", "javaVersion") |
| |
| extractMatchingApks = pctx.StaticRule( |
| "extractMatchingApks", |
| blueprint.RuleParams{ |
| Command: `rm -rf "$out" && ` + |
| `${config.ExtractApksCmd} -o "${out}" -zip "${zip}" -allow-prereleased=${allow-prereleased} ` + |
| `-sdk-version=${sdk-version} -skip-sdk-check=${skip-sdk-check} -abis=${abis} ` + |
| `--screen-densities=${screen-densities} --stem=${stem} ` + |
| `-apkcerts=${apkcerts} -partition=${partition} ` + |
| `${in}`, |
| CommandDeps: []string{"${config.ExtractApksCmd}"}, |
| }, |
| "abis", "allow-prereleased", "screen-densities", "sdk-version", "skip-sdk-check", "stem", "apkcerts", "partition", "zip") |
| |
| turbine, turbineRE = pctx.RemoteStaticRules("turbine", |
| blueprint.RuleParams{ |
| Command: `$reTemplate${config.JavaCmd} ${config.JavaVmFlags} -jar ${config.TurbineJar} $outputFlags ` + |
| `--sources @$out.rsp --source_jars $srcJars ` + |
| `--javacopts ${config.CommonJdkFlags} ` + |
| `$javacFlags -source $javaVersion -target $javaVersion -- $turbineFlags && ` + |
| `(for o in $outputs; do if cmp -s $${o}.tmp $${o} ; then rm $${o}.tmp ; else mv $${o}.tmp $${o} ; fi; done )`, |
| CommandDeps: []string{ |
| "${config.TurbineJar}", |
| "${config.JavaCmd}", |
| }, |
| Rspfile: "$out.rsp", |
| RspfileContent: "$in", |
| Restat: true, |
| }, |
| &remoteexec.REParams{Labels: map[string]string{"type": "tool", "name": "turbine"}, |
| ExecStrategy: "${config.RETurbineExecStrategy}", |
| Inputs: []string{"${config.TurbineJar}", "${out}.rsp", "$implicits"}, |
| RSPFiles: []string{"${out}.rsp"}, |
| OutputFiles: []string{"$rbeOutputs"}, |
| ToolchainInputs: []string{"${config.JavaCmd}"}, |
| Platform: map[string]string{remoteexec.PoolKey: "${config.REJavaPool}"}, |
| }, |
| []string{"javacFlags", "turbineFlags", "outputFlags", "javaVersion", "outputs", "rbeOutputs", "srcJars"}, []string{"implicits"}) |
| |
| jar, jarRE = pctx.RemoteStaticRules("jar", |
| blueprint.RuleParams{ |
| Command: `$reTemplate${config.SoongZipCmd} -jar -o $out @$out.rsp`, |
| CommandDeps: []string{"${config.SoongZipCmd}"}, |
| Rspfile: "$out.rsp", |
| RspfileContent: "$jarArgs", |
| }, |
| &remoteexec.REParams{ |
| ExecStrategy: "${config.REJarExecStrategy}", |
| Inputs: []string{"${config.SoongZipCmd}", "${out}.rsp"}, |
| RSPFiles: []string{"${out}.rsp"}, |
| OutputFiles: []string{"$out"}, |
| Platform: map[string]string{remoteexec.PoolKey: "${config.REJavaPool}"}, |
| }, []string{"jarArgs"}, nil) |
| |
| zip, zipRE = pctx.RemoteStaticRules("zip", |
| blueprint.RuleParams{ |
| Command: `${config.SoongZipCmd} -o $out @$out.rsp`, |
| CommandDeps: []string{"${config.SoongZipCmd}"}, |
| Rspfile: "$out.rsp", |
| RspfileContent: "$jarArgs", |
| }, |
| &remoteexec.REParams{ |
| ExecStrategy: "${config.REZipExecStrategy}", |
| Inputs: []string{"${config.SoongZipCmd}", "${out}.rsp", "$implicits"}, |
| RSPFiles: []string{"${out}.rsp"}, |
| OutputFiles: []string{"$out"}, |
| Platform: map[string]string{remoteexec.PoolKey: "${config.REJavaPool}"}, |
| }, []string{"jarArgs"}, []string{"implicits"}) |
| |
| combineJar = pctx.AndroidStaticRule("combineJar", |
| blueprint.RuleParams{ |
| Command: `${config.MergeZipsCmd} --ignore-duplicates -j $jarArgs $out $in`, |
| CommandDeps: []string{"${config.MergeZipsCmd}"}, |
| }, |
| "jarArgs") |
| |
| jarjar = pctx.AndroidStaticRule("jarjar", |
| blueprint.RuleParams{ |
| Command: "" + |
| // Jarjar doesn't exit with an error when the rules file contains a syntax error, |
| // leading to stale or missing files later in the build. Remove the output file |
| // before running jarjar. |
| "rm -f ${out} && " + |
| "${config.JavaCmd} ${config.JavaVmFlags}" + |
| // b/146418363 Enable Android specific jarjar transformer to drop compat annotations |
| // for newly repackaged classes. Dropping @UnsupportedAppUsage on repackaged classes |
| // avoids adding new hiddenapis after jarjar'ing. |
| " -DremoveAndroidCompatAnnotations=true" + |
| " -jar ${config.JarjarCmd} process $rulesFile $in $out && " + |
| // Turn a missing output file into a ninja error |
| `[ -e ${out} ] || (echo "Missing output file"; exit 1)`, |
| CommandDeps: []string{"${config.JavaCmd}", "${config.JarjarCmd}", "$rulesFile"}, |
| }, |
| "rulesFile") |
| |
| packageCheck = pctx.AndroidStaticRule("packageCheck", |
| blueprint.RuleParams{ |
| Command: "rm -f $out && " + |
| "${config.PackageCheckCmd} $in $packages && " + |
| "touch $out", |
| CommandDeps: []string{"${config.PackageCheckCmd}"}, |
| }, |
| "packages") |
| |
| jetifier = pctx.AndroidStaticRule("jetifier", |
| blueprint.RuleParams{ |
| Command: "${config.JavaCmd} ${config.JavaVmFlags} -jar ${config.JetifierJar} -l error -o $out -i $in -t epoch", |
| CommandDeps: []string{"${config.JavaCmd}", "${config.JetifierJar}"}, |
| }, |
| ) |
| |
| zipalign = pctx.AndroidStaticRule("zipalign", |
| blueprint.RuleParams{ |
| Command: "if ! ${config.ZipAlign} -c -p 4 $in > /dev/null; then " + |
| "${config.ZipAlign} -f -p 4 $in $out; " + |
| "else " + |
| "cp -f $in $out; " + |
| "fi", |
| CommandDeps: []string{"${config.ZipAlign}"}, |
| }, |
| ) |
| |
| convertImplementationJarToHeaderJarRule = pctx.AndroidStaticRule("convertImplementationJarToHeaderJar", |
| blueprint.RuleParams{ |
| Command: `${config.Zip2ZipCmd} -i ${in} -o ${out} -x 'META-INF/services/**/*'`, |
| CommandDeps: []string{"${config.Zip2ZipCmd}"}, |
| }) |
| |
| writeCombinedProguardFlagsFileRule = pctx.AndroidStaticRule("writeCombinedProguardFlagsFileRule", |
| blueprint.RuleParams{ |
| Command: `rm -f $out && ` + |
| `for f in $in; do ` + |
| ` echo && ` + |
| ` echo "# including $$f" && ` + |
| ` cat $$f; ` + |
| `done > $out`, |
| }) |
| |
| gatherReleasedFlaggedApisRule = pctx.AndroidStaticRule("gatherReleasedFlaggedApisRule", |
| blueprint.RuleParams{ |
| Command: `${aconfig} dump-cache --format='{fully_qualified_name}={state:bool}' ` + |
| `--out ${out} ` + |
| `${flags_path} ` + |
| `${filter_args} `, |
| CommandDeps: []string{"${aconfig}"}, |
| Description: "aconfig_bool", |
| }, "flags_path", "filter_args") |
| |
| generateMetalavaRevertAnnotationsRule = pctx.AndroidStaticRule("generateMetalavaRevertAnnotationsRule", |
| blueprint.RuleParams{ |
| Command: `${keep-flagged-apis} ${in} > ${out}`, |
| CommandDeps: []string{"${keep-flagged-apis}"}, |
| }) |
| ) |
| |
| func init() { |
| pctx.Import("android/soong/android") |
| pctx.Import("android/soong/java/config") |
| |
| pctx.HostBinToolVariable("aconfig", "aconfig") |
| pctx.HostBinToolVariable("keep-flagged-apis", "keep-flagged-apis") |
| } |
| |
| type javaBuilderFlags struct { |
| javacFlags string |
| |
| // bootClasspath is the list of jars that form the boot classpath (generally the java.* and |
| // android.* classes) for tools that still use it. javac targeting 1.9 or higher uses |
| // systemModules and java9Classpath instead. |
| bootClasspath classpath |
| |
| // classpath is the list of jars that form the classpath for javac and kotlinc rules. It |
| // contains header jars for all static and non-static dependencies. |
| classpath classpath |
| |
| // dexClasspath is the list of jars that form the classpath for d8 and r8 rules. It contains |
| // header jars for all non-static dependencies. Static dependencies have already been |
| // combined into the program jar. |
| dexClasspath classpath |
| |
| // java9Classpath is the list of jars that will be added to the classpath when targeting |
| // 1.9 or higher. It generally contains the android.* classes, while the java.* classes |
| // are provided by systemModules. |
| java9Classpath classpath |
| |
| processorPath classpath |
| processors []string |
| systemModules *systemModules |
| aidlFlags string |
| aidlDeps android.Paths |
| javaVersion javaVersion |
| |
| errorProneExtraJavacFlags string |
| errorProneProcessorPath classpath |
| |
| kotlincFlags string |
| kotlincClasspath classpath |
| kotlincDeps android.Paths |
| |
| proto android.ProtoFlags |
| } |
| |
| func DefaultJavaBuilderFlags() javaBuilderFlags { |
| return javaBuilderFlags{ |
| javaVersion: JAVA_VERSION_8, |
| } |
| } |
| |
| func TransformJavaToClasses(ctx android.ModuleContext, outputFile android.WritablePath, shardIdx int, |
| srcFiles, srcJars android.Paths, annoSrcJar android.WritablePath, flags javaBuilderFlags, deps android.Paths) { |
| |
| // Compile java sources into .class files |
| desc := "javac" |
| if shardIdx >= 0 { |
| desc += strconv.Itoa(shardIdx) |
| } |
| |
| transformJavaToClasses(ctx, outputFile, shardIdx, srcFiles, srcJars, annoSrcJar, flags, deps, "javac", desc) |
| } |
| |
| // Emits the rule to generate Xref input file (.kzip file) for the given set of source files and source jars |
| // to compile with given set of builder flags, etc. |
| func emitXrefRule(ctx android.ModuleContext, xrefFile android.WritablePath, idx int, |
| srcFiles, srcJars android.Paths, |
| flags javaBuilderFlags, deps android.Paths) { |
| |
| deps = append(deps, srcJars...) |
| classpath := flags.classpath |
| |
| var bootClasspath string |
| if flags.javaVersion.usesJavaModules() { |
| var systemModuleDeps android.Paths |
| bootClasspath, systemModuleDeps = flags.systemModules.FormJavaSystemModulesPath(ctx.Device()) |
| deps = append(deps, systemModuleDeps...) |
| classpath = append(flags.java9Classpath, classpath...) |
| } else { |
| deps = append(deps, flags.bootClasspath...) |
| if len(flags.bootClasspath) == 0 && ctx.Device() { |
| // explicitly specify -bootclasspath "" if the bootclasspath is empty to |
| // ensure java does not fall back to the default bootclasspath. |
| bootClasspath = `-bootclasspath ""` |
| } else { |
| bootClasspath = flags.bootClasspath.FormJavaClassPath("-bootclasspath") |
| } |
| } |
| |
| deps = append(deps, classpath...) |
| deps = append(deps, flags.processorPath...) |
| |
| processor := "-proc:none" |
| if len(flags.processors) > 0 { |
| processor = "-processor " + strings.Join(flags.processors, ",") |
| } |
| |
| intermediatesDir := "xref" |
| if idx >= 0 { |
| intermediatesDir += strconv.Itoa(idx) |
| } |
| |
| ctx.Build(pctx, |
| android.BuildParams{ |
| Rule: kytheExtract, |
| Description: "Xref Java extractor", |
| Output: xrefFile, |
| Inputs: srcFiles, |
| Implicits: deps, |
| Args: map[string]string{ |
| "annoDir": android.PathForModuleOut(ctx, intermediatesDir, "anno").String(), |
| "bootClasspath": bootClasspath, |
| "classpath": classpath.FormJavaClassPath("-classpath"), |
| "javacFlags": flags.javacFlags, |
| "javaVersion": flags.javaVersion.String(), |
| "outDir": android.PathForModuleOut(ctx, "javac", "classes.xref").String(), |
| "processorpath": flags.processorPath.FormJavaClassPath("-processorpath"), |
| "processor": processor, |
| "srcJarDir": android.PathForModuleOut(ctx, intermediatesDir, "srcjars.xref").String(), |
| "srcJars": strings.Join(srcJars.Strings(), " "), |
| }, |
| }) |
| } |
| |
| func turbineFlags(ctx android.ModuleContext, flags javaBuilderFlags) (string, android.Paths) { |
| var deps android.Paths |
| |
| classpath := flags.classpath |
| |
| var bootClasspath string |
| if flags.javaVersion.usesJavaModules() { |
| var systemModuleDeps android.Paths |
| bootClasspath, systemModuleDeps = flags.systemModules.FormTurbineSystemModulesPath(ctx.Device()) |
| deps = append(deps, systemModuleDeps...) |
| classpath = append(flags.java9Classpath, classpath...) |
| } else { |
| deps = append(deps, flags.bootClasspath...) |
| if len(flags.bootClasspath) == 0 && ctx.Device() { |
| // explicitly specify -bootclasspath "" if the bootclasspath is empty to |
| // ensure turbine does not fall back to the default bootclasspath. |
| bootClasspath = `--bootclasspath ""` |
| } else { |
| bootClasspath = flags.bootClasspath.FormTurbineClassPath("--bootclasspath ") |
| } |
| } |
| |
| deps = append(deps, classpath...) |
| turbineFlags := bootClasspath + " " + classpath.FormTurbineClassPath("--classpath ") |
| |
| return turbineFlags, deps |
| } |
| |
| func TransformJavaToHeaderClasses(ctx android.ModuleContext, outputFile android.WritablePath, |
| srcFiles, srcJars android.Paths, flags javaBuilderFlags) { |
| |
| turbineFlags, deps := turbineFlags(ctx, flags) |
| |
| deps = append(deps, srcJars...) |
| |
| rule := turbine |
| args := map[string]string{ |
| "javacFlags": flags.javacFlags, |
| "srcJars": strings.Join(srcJars.Strings(), " "), |
| "javaVersion": flags.javaVersion.String(), |
| "turbineFlags": turbineFlags, |
| "outputFlags": "--output " + outputFile.String() + ".tmp", |
| "outputs": outputFile.String(), |
| } |
| if ctx.Config().UseRBE() && ctx.Config().IsEnvTrue("RBE_TURBINE") { |
| rule = turbineRE |
| args["implicits"] = strings.Join(deps.Strings(), ",") |
| args["rbeOutputs"] = outputFile.String() + ".tmp" |
| } |
| ctx.Build(pctx, android.BuildParams{ |
| Rule: rule, |
| Description: "turbine", |
| Output: outputFile, |
| Inputs: srcFiles, |
| Implicits: deps, |
| Args: args, |
| }) |
| } |
| |
| // TurbineApt produces a rule to run annotation processors using turbine. |
| func TurbineApt(ctx android.ModuleContext, outputSrcJar, outputResJar android.WritablePath, |
| srcFiles, srcJars android.Paths, flags javaBuilderFlags) { |
| |
| turbineFlags, deps := turbineFlags(ctx, flags) |
| |
| deps = append(deps, srcJars...) |
| |
| deps = append(deps, flags.processorPath...) |
| turbineFlags += " " + flags.processorPath.FormTurbineClassPath("--processorpath ") |
| turbineFlags += " --processors " + strings.Join(flags.processors, " ") |
| |
| outputs := android.WritablePaths{outputSrcJar, outputResJar} |
| outputFlags := "--gensrc_output " + outputSrcJar.String() + ".tmp " + |
| "--resource_output " + outputResJar.String() + ".tmp" |
| |
| rule := turbine |
| args := map[string]string{ |
| "javacFlags": flags.javacFlags, |
| "srcJars": strings.Join(srcJars.Strings(), " "), |
| "javaVersion": flags.javaVersion.String(), |
| "turbineFlags": turbineFlags, |
| "outputFlags": outputFlags, |
| "outputs": strings.Join(outputs.Strings(), " "), |
| } |
| if ctx.Config().UseRBE() && ctx.Config().IsEnvTrue("RBE_TURBINE") { |
| rule = turbineRE |
| args["implicits"] = strings.Join(deps.Strings(), ",") |
| args["rbeOutputs"] = outputSrcJar.String() + ".tmp," + outputResJar.String() + ".tmp" |
| } |
| ctx.Build(pctx, android.BuildParams{ |
| Rule: rule, |
| Description: "turbine apt", |
| Output: outputs[0], |
| ImplicitOutputs: outputs[1:], |
| Inputs: srcFiles, |
| Implicits: deps, |
| Args: args, |
| }) |
| } |
| |
| // transformJavaToClasses takes source files and converts them to a jar containing .class files. |
| // srcFiles is a list of paths to sources, srcJars is a list of paths to jar files that contain |
| // sources. flags contains various command line flags to be passed to the compiler. |
| // |
| // This method may be used for different compilers, including javac and Error Prone. The rule |
| // argument specifies which command line to use and desc sets the description of the rule that will |
| // be printed at build time. The stem argument provides the file name of the output jar, and |
| // suffix will be appended to various intermediate files and directories to avoid collisions when |
| // this function is called twice in the same module directory. |
| func transformJavaToClasses(ctx android.ModuleContext, outputFile android.WritablePath, |
| shardIdx int, srcFiles, srcJars android.Paths, annoSrcJar android.WritablePath, |
| flags javaBuilderFlags, deps android.Paths, |
| intermediatesDir, desc string) { |
| |
| deps = append(deps, srcJars...) |
| |
| classpath := flags.classpath |
| |
| var bootClasspath string |
| if flags.javaVersion.usesJavaModules() { |
| var systemModuleDeps android.Paths |
| bootClasspath, systemModuleDeps = flags.systemModules.FormJavaSystemModulesPath(ctx.Device()) |
| deps = append(deps, systemModuleDeps...) |
| classpath = append(flags.java9Classpath, classpath...) |
| } else { |
| deps = append(deps, flags.bootClasspath...) |
| if len(flags.bootClasspath) == 0 && ctx.Device() { |
| // explicitly specify -bootclasspath "" if the bootclasspath is empty to |
| // ensure java does not fall back to the default bootclasspath. |
| bootClasspath = `-bootclasspath ""` |
| } else { |
| bootClasspath = flags.bootClasspath.FormJavaClassPath("-bootclasspath") |
| } |
| } |
| |
| deps = append(deps, classpath...) |
| deps = append(deps, flags.processorPath...) |
| |
| processor := "-proc:none" |
| if len(flags.processors) > 0 { |
| processor = "-processor " + strings.Join(flags.processors, ",") |
| } |
| |
| srcJarDir := "srcjars" |
| outDir := "classes" |
| annoDir := "anno" |
| if shardIdx >= 0 { |
| shardDir := "shard" + strconv.Itoa(shardIdx) |
| srcJarDir = filepath.Join(shardDir, srcJarDir) |
| outDir = filepath.Join(shardDir, outDir) |
| annoDir = filepath.Join(shardDir, annoDir) |
| } |
| rule := javac |
| if ctx.Config().UseRBE() && ctx.Config().IsEnvTrue("RBE_JAVAC") { |
| rule = javacRE |
| } |
| ctx.Build(pctx, android.BuildParams{ |
| Rule: rule, |
| Description: desc, |
| Output: outputFile, |
| ImplicitOutput: annoSrcJar, |
| Inputs: srcFiles, |
| Implicits: deps, |
| Args: map[string]string{ |
| "javacFlags": flags.javacFlags, |
| "bootClasspath": bootClasspath, |
| "classpath": classpath.FormJavaClassPath("-classpath"), |
| "processorpath": flags.processorPath.FormJavaClassPath("-processorpath"), |
| "processor": processor, |
| "srcJars": strings.Join(srcJars.Strings(), " "), |
| "srcJarDir": android.PathForModuleOut(ctx, intermediatesDir, srcJarDir).String(), |
| "outDir": android.PathForModuleOut(ctx, intermediatesDir, outDir).String(), |
| "annoDir": android.PathForModuleOut(ctx, intermediatesDir, annoDir).String(), |
| "annoSrcJar": annoSrcJar.String(), |
| "javaVersion": flags.javaVersion.String(), |
| }, |
| }) |
| } |
| |
| func TransformResourcesToJar(ctx android.ModuleContext, outputFile android.WritablePath, |
| jarArgs []string, deps android.Paths) { |
| |
| rule := jar |
| if ctx.Config().UseRBE() && ctx.Config().IsEnvTrue("RBE_JAR") { |
| rule = jarRE |
| } |
| ctx.Build(pctx, android.BuildParams{ |
| Rule: rule, |
| Description: "jar", |
| Output: outputFile, |
| Implicits: deps, |
| Args: map[string]string{ |
| "jarArgs": strings.Join(proptools.NinjaAndShellEscapeList(jarArgs), " "), |
| }, |
| }) |
| } |
| |
| func TransformJarsToJar(ctx android.ModuleContext, outputFile android.WritablePath, desc string, |
| jars android.Paths, manifest android.OptionalPath, stripDirEntries bool, filesToStrip []string, |
| dirsToStrip []string) { |
| |
| var deps android.Paths |
| |
| var jarArgs []string |
| if manifest.Valid() { |
| jarArgs = append(jarArgs, "-m ", manifest.String()) |
| deps = append(deps, manifest.Path()) |
| } |
| |
| for _, dir := range dirsToStrip { |
| jarArgs = append(jarArgs, "-stripDir ", dir) |
| } |
| |
| for _, file := range filesToStrip { |
| jarArgs = append(jarArgs, "-stripFile ", file) |
| } |
| |
| // Remove any module-info.class files that may have come from prebuilt jars, they cause problems |
| // for downstream tools like desugar. |
| jarArgs = append(jarArgs, "-stripFile module-info.class") |
| |
| if stripDirEntries { |
| jarArgs = append(jarArgs, "-D") |
| } |
| |
| ctx.Build(pctx, android.BuildParams{ |
| Rule: combineJar, |
| Description: desc, |
| Output: outputFile, |
| Inputs: jars, |
| Implicits: deps, |
| Args: map[string]string{ |
| "jarArgs": strings.Join(jarArgs, " "), |
| }, |
| }) |
| } |
| |
| func convertImplementationJarToHeaderJar(ctx android.ModuleContext, implementationJarFile android.Path, |
| headerJarFile android.WritablePath) { |
| ctx.Build(pctx, android.BuildParams{ |
| Rule: convertImplementationJarToHeaderJarRule, |
| Input: implementationJarFile, |
| Output: headerJarFile, |
| }) |
| } |
| |
| func TransformJarJar(ctx android.ModuleContext, outputFile android.WritablePath, |
| classesJar android.Path, rulesFile android.Path) { |
| ctx.Build(pctx, android.BuildParams{ |
| Rule: jarjar, |
| Description: "jarjar", |
| Output: outputFile, |
| Input: classesJar, |
| Implicit: rulesFile, |
| Args: map[string]string{ |
| "rulesFile": rulesFile.String(), |
| }, |
| }) |
| } |
| |
| func CheckJarPackages(ctx android.ModuleContext, outputFile android.WritablePath, |
| classesJar android.Path, permittedPackages []string) { |
| ctx.Build(pctx, android.BuildParams{ |
| Rule: packageCheck, |
| Description: "packageCheck", |
| Output: outputFile, |
| Input: classesJar, |
| Args: map[string]string{ |
| "packages": strings.Join(permittedPackages, " "), |
| }, |
| }) |
| } |
| |
| func TransformJetifier(ctx android.ModuleContext, outputFile android.WritablePath, |
| inputFile android.Path) { |
| ctx.Build(pctx, android.BuildParams{ |
| Rule: jetifier, |
| Description: "jetifier", |
| Output: outputFile, |
| Input: inputFile, |
| }) |
| } |
| |
| func GenerateMainClassManifest(ctx android.ModuleContext, outputFile android.WritablePath, mainClass string) { |
| android.WriteFileRule(ctx, outputFile, "Main-Class: "+mainClass+"\n") |
| } |
| |
| func TransformZipAlign(ctx android.ModuleContext, outputFile android.WritablePath, inputFile android.Path, validations android.Paths) { |
| ctx.Build(pctx, android.BuildParams{ |
| Rule: zipalign, |
| Description: "align", |
| Input: inputFile, |
| Output: outputFile, |
| Validations: validations, |
| }) |
| } |
| |
| func writeCombinedProguardFlagsFile(ctx android.ModuleContext, outputFile android.WritablePath, files android.Paths) { |
| ctx.Build(pctx, android.BuildParams{ |
| Rule: writeCombinedProguardFlagsFileRule, |
| Description: "write combined proguard flags file", |
| Inputs: files, |
| Output: outputFile, |
| }) |
| } |
| |
| type classpath android.Paths |
| |
| func (x *classpath) formJoinedClassPath(optName string, sep string) string { |
| if optName != "" && !strings.HasSuffix(optName, "=") && !strings.HasSuffix(optName, " ") { |
| optName += " " |
| } |
| if len(*x) > 0 { |
| return optName + strings.Join(x.Strings(), sep) |
| } else { |
| return "" |
| } |
| } |
| func (x *classpath) FormJavaClassPath(optName string) string { |
| return x.formJoinedClassPath(optName, ":") |
| } |
| |
| func (x *classpath) FormTurbineClassPath(optName string) string { |
| return x.formJoinedClassPath(optName, " ") |
| } |
| |
| // FormRepeatedClassPath returns a list of arguments with the given optName prefixed to each element of the classpath. |
| func (x *classpath) FormRepeatedClassPath(optName string) []string { |
| if x == nil || *x == nil { |
| return nil |
| } |
| flags := make([]string, len(*x)) |
| for i, v := range *x { |
| flags[i] = optName + v.String() |
| } |
| |
| return flags |
| } |
| |
| // Convert a classpath to an android.Paths |
| func (x *classpath) Paths() android.Paths { |
| return append(android.Paths(nil), (*x)...) |
| } |
| |
| func (x *classpath) Strings() []string { |
| if x == nil { |
| return nil |
| } |
| ret := make([]string, len(*x)) |
| for i, path := range *x { |
| ret[i] = path.String() |
| } |
| return ret |
| } |
| |
| type systemModules struct { |
| dir android.Path |
| deps android.Paths |
| } |
| |
| // Returns a --system argument in the form javac expects with -source 1.9 and the list of files to |
| // depend on. If forceEmpty is true, returns --system=none if the list is empty to ensure javac |
| // does not fall back to the default system modules. |
| func (x *systemModules) FormJavaSystemModulesPath(forceEmpty bool) (string, android.Paths) { |
| if x != nil { |
| return "--system=" + x.dir.String(), x.deps |
| } else if forceEmpty { |
| return "--system=none", nil |
| } else { |
| return "", nil |
| } |
| } |
| |
| // Returns a --system argument in the form turbine expects with -source 1.9 and the list of files to |
| // depend on. If forceEmpty is true, returns --bootclasspath "" if the list is empty to ensure turbine |
| // does not fall back to the default bootclasspath. |
| func (x *systemModules) FormTurbineSystemModulesPath(forceEmpty bool) (string, android.Paths) { |
| if x != nil { |
| return "--system " + x.dir.String(), x.deps |
| } else if forceEmpty { |
| return `--bootclasspath ""`, nil |
| } else { |
| return "--system ${config.JavaHome}", nil |
| } |
| } |