79b7ca1e7a
Change-Id: Ie977f08f8aee59ea21dcbfe7c9c07dc0dc535775 Fixes: 328877412 Test: TH
850 lines
31 KiB
Go
850 lines
31 KiB
Go
// 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 ` +
|
|
`--add-exports=jdk.compiler/com.sun.tools.javac.tree=ALL-UNNAMED ` +
|
|
`--add-exports=jdk.internal.opt/jdk.internal.opt=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")
|
|
combineJarRsp = pctx.AndroidStaticRule("combineJarRsp",
|
|
blueprint.RuleParams{
|
|
Command: `${config.MergeZipsCmd} --ignore-duplicates -j $jarArgs $out @$out.rsp`,
|
|
CommandDeps: []string{"${config.MergeZipsCmd}"},
|
|
Rspfile: "$out.rsp",
|
|
RspfileContent: "$in",
|
|
},
|
|
"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 --dedup --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, dir string) (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 ")
|
|
|
|
const flagsLimit = 32 * 1024
|
|
if len(turbineFlags) > flagsLimit {
|
|
flagsRspFile := android.PathForModuleOut(ctx, dir, "turbine-flags.rsp")
|
|
android.WriteFileRule(ctx, flagsRspFile, turbineFlags)
|
|
turbineFlags = "@" + flagsRspFile.String()
|
|
deps = append(deps, flagsRspFile)
|
|
}
|
|
|
|
return turbineFlags, deps
|
|
}
|
|
|
|
func TransformJavaToHeaderClasses(ctx android.ModuleContext, outputFile android.WritablePath,
|
|
srcFiles, srcJars android.Paths, flags javaBuilderFlags) {
|
|
|
|
turbineFlags, deps := turbineFlags(ctx, flags, "turbine")
|
|
|
|
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, "kapt")
|
|
|
|
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...)
|
|
|
|
javacClasspath := flags.classpath
|
|
|
|
var bootClasspath string
|
|
if flags.javaVersion.usesJavaModules() {
|
|
var systemModuleDeps android.Paths
|
|
bootClasspath, systemModuleDeps = flags.systemModules.FormJavaSystemModulesPath(ctx.Device())
|
|
deps = append(deps, systemModuleDeps...)
|
|
javacClasspath = append(flags.java9Classpath, javacClasspath...)
|
|
} 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")
|
|
}
|
|
}
|
|
|
|
classpathArg := javacClasspath.FormJavaClassPath("-classpath")
|
|
|
|
// Keep the command line under the MAX_ARG_STRLEN limit by putting the classpath argument into an rsp file
|
|
// if it is too long.
|
|
const classpathLimit = 64 * 1024
|
|
if len(classpathArg) > classpathLimit {
|
|
classpathRspFile := outputFile.ReplaceExtension(ctx, "classpath")
|
|
android.WriteFileRule(ctx, classpathRspFile, classpathArg)
|
|
deps = append(deps, classpathRspFile)
|
|
classpathArg = "@" + classpathRspFile.String()
|
|
}
|
|
|
|
deps = append(deps, javacClasspath...)
|
|
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": classpathArg,
|
|
"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")
|
|
}
|
|
|
|
rule := combineJar
|
|
// Keep the command line under the MAX_ARG_STRLEN limit by putting the list of jars into an rsp file
|
|
// if it is too long.
|
|
const jarsLengthLimit = 64 * 1024
|
|
jarsLength := 0
|
|
for i, jar := range jars {
|
|
if i != 0 {
|
|
jarsLength += 1
|
|
}
|
|
jarsLength += len(jar.String())
|
|
}
|
|
if jarsLength > jarsLengthLimit {
|
|
rule = combineJarRsp
|
|
}
|
|
|
|
ctx.Build(pctx, android.BuildParams{
|
|
Rule: rule,
|
|
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
|
|
}
|
|
}
|