// 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 ( "strings" "github.com/google/blueprint" "android/soong/android" "android/soong/java/config" ) 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 a post-processing step lists // the files in the the directory into a list file that later rules depend on (and sometimes // read from directly using @) javac = pctx.AndroidGomaStaticRule("javac", blueprint.RuleParams{ Command: `rm -rf "$outDir" "$annoDir" && mkdir -p "$outDir" "$annoDir" && ` + `${config.JavacWrapper}${config.JavacCmd} ${config.JavacHeapFlags} ${config.CommonJdkFlags} ` + `$javacFlags $bootClasspath $classpath ` + `-source $javaVersion -target $javaVersion ` + `-d $outDir -s $annoDir @$out.rsp && ` + `${config.SoongZipCmd} -jar -o $out -C $outDir -D $outDir`, CommandDeps: []string{"${config.JavacCmd}", "${config.SoongZipCmd}"}, Rspfile: "$out.rsp", RspfileContent: "$in", }, "javacFlags", "bootClasspath", "classpath", "outDir", "annoDir", "javaVersion") errorprone = pctx.AndroidStaticRule("errorprone", blueprint.RuleParams{ Command: `rm -rf "$outDir" "$annoDir" && mkdir -p "$outDir" "$annoDir" && ` + `${config.ErrorProneCmd}` + `$javacFlags $bootClasspath $classpath ` + `-source $javaVersion -target $javaVersion ` + `-d $outDir -s $annoDir @$out.rsp && ` + `${config.SoongZipCmd} -jar -o $out -C $outDir -D $outDir`, CommandDeps: []string{ "${config.JavaCmd}", "${config.ErrorProneJavacJar}", "${config.ErrorProneJar}", "${config.SoongZipCmd}", }, Rspfile: "$out.rsp", RspfileContent: "$in", }, "javacFlags", "bootClasspath", "classpath", "outDir", "annoDir", "javaVersion") jar = pctx.AndroidStaticRule("jar", blueprint.RuleParams{ Command: `${config.SoongZipCmd} -jar -o $out $jarArgs`, CommandDeps: []string{"${config.SoongZipCmd}"}, }, "jarArgs") combineJar = pctx.AndroidStaticRule("combineJar", blueprint.RuleParams{ Command: `${config.MergeZipsCmd} -j $out $in`, CommandDeps: []string{"${config.MergeZipsCmd}"}, }, "outDir") dx = pctx.AndroidStaticRule("dx", blueprint.RuleParams{ Command: `rm -rf "$outDir" && mkdir -p "$outDir" && ` + `${config.DxCmd} --dex --output=$outDir $dxFlags $in || ( rm -rf "$outDir"; exit 41 ) && ` + `find "$outDir" -name "classes*.dex" | sort > $out`, CommandDeps: []string{"${config.DxCmd}"}, }, "outDir", "dxFlags") jarjar = pctx.AndroidStaticRule("jarjar", blueprint.RuleParams{ Command: "${config.JavaCmd} -jar ${config.JarjarCmd} process $rulesFile $in $out", CommandDeps: []string{"${config.JavaCmd}", "${config.JarjarCmd}", "$rulesFile"}, }, "rulesFile") ) func init() { pctx.Import("android/soong/java/config") } type javaBuilderFlags struct { javacFlags string dxFlags string bootClasspath string classpath string aidlFlags string javaVersion string } type jarSpec struct { fileList, dir android.Path } func (j jarSpec) soongJarArgs() string { return "-C " + j.dir.String() + " -l " + j.fileList.String() } func TransformJavaToClasses(ctx android.ModuleContext, srcFiles, srcFileLists android.Paths, flags javaBuilderFlags, deps android.Paths) android.ModuleOutPath { classDir := android.PathForModuleOut(ctx, "classes") annoDir := android.PathForModuleOut(ctx, "anno") classJar := android.PathForModuleOut(ctx, "classes.jar") javacFlags := flags.javacFlags + android.JoinWithPrefix(srcFileLists.Strings(), "@") deps = append(deps, srcFileLists...) ctx.ModuleBuild(pctx, android.ModuleBuildParams{ Rule: javac, Description: "javac", Output: classJar, Inputs: srcFiles, Implicits: deps, Args: map[string]string{ "javacFlags": javacFlags, "bootClasspath": flags.bootClasspath, "classpath": flags.classpath, "outDir": classDir.String(), "annoDir": annoDir.String(), "javaVersion": flags.javaVersion, }, }) return classJar } func RunErrorProne(ctx android.ModuleContext, srcFiles android.Paths, srcFileLists android.Paths, flags javaBuilderFlags, deps android.Paths) android.Path { if config.ErrorProneJar == "" { ctx.ModuleErrorf("cannot build with Error Prone, missing external/error_prone?") return nil } classDir := android.PathForModuleOut(ctx, "classes-errorprone") annoDir := android.PathForModuleOut(ctx, "anno-errorprone") classFileList := android.PathForModuleOut(ctx, "classes-errorprone.list") javacFlags := flags.javacFlags + android.JoinWithPrefix(srcFileLists.Strings(), "@") deps = append(deps, srcFileLists...) ctx.ModuleBuild(pctx, android.ModuleBuildParams{ Rule: errorprone, Description: "errorprone", Output: classFileList, Inputs: srcFiles, Implicits: deps, Args: map[string]string{ "javacFlags": javacFlags, "bootClasspath": flags.bootClasspath, "classpath": flags.classpath, "outDir": classDir.String(), "annoDir": annoDir.String(), "javaVersion": flags.javaVersion, }, }) return classFileList } func TransformResourcesToJar(ctx android.ModuleContext, resources []jarSpec, manifest android.OptionalPath, deps android.Paths) android.Path { outputFile := android.PathForModuleOut(ctx, "res.jar") jarArgs := []string{} for _, j := range resources { deps = append(deps, j.fileList) jarArgs = append(jarArgs, j.soongJarArgs()) } if manifest.Valid() { deps = append(deps, manifest.Path()) jarArgs = append(jarArgs, "-m "+manifest.String()) } ctx.ModuleBuild(pctx, android.ModuleBuildParams{ Rule: jar, Description: "jar", Output: outputFile, Implicits: deps, Args: map[string]string{ "jarArgs": strings.Join(jarArgs, " "), }, }) return outputFile } func TransformJarsToJar(ctx android.ModuleContext, stem string, jars android.Paths) android.Path { outputFile := android.PathForModuleOut(ctx, stem) if len(jars) == 1 { return jars[0] } ctx.ModuleBuild(pctx, android.ModuleBuildParams{ Rule: combineJar, Description: "combine jars", Output: outputFile, Inputs: jars, }) return outputFile } func TransformClassesJarToDex(ctx android.ModuleContext, classesJar android.Path, flags javaBuilderFlags) jarSpec { outDir := android.PathForModuleOut(ctx, "dex") outputFile := android.PathForModuleOut(ctx, "dex.filelist") ctx.ModuleBuild(pctx, android.ModuleBuildParams{ Rule: dx, Description: "dx", Output: outputFile, Input: classesJar, Args: map[string]string{ "dxFlags": flags.dxFlags, "outDir": outDir.String(), }, }) return jarSpec{outputFile, outDir} } func TransformDexToJavaLib(ctx android.ModuleContext, resources []jarSpec, dexJarSpec jarSpec) android.Path { outputFile := android.PathForModuleOut(ctx, "javalib.jar") var deps android.Paths var jarArgs []string for _, j := range resources { deps = append(deps, j.fileList) jarArgs = append(jarArgs, j.soongJarArgs()) } deps = append(deps, dexJarSpec.fileList) jarArgs = append(jarArgs, dexJarSpec.soongJarArgs()) ctx.ModuleBuild(pctx, android.ModuleBuildParams{ Rule: jar, Description: "jar", Output: outputFile, Implicits: deps, Args: map[string]string{ "jarArgs": strings.Join(jarArgs, " "), }, }) return outputFile } func TransformJarJar(ctx android.ModuleContext, classesJar android.Path, rulesFile android.Path) android.ModuleOutPath { outputFile := android.PathForModuleOut(ctx, "classes-jarjar.jar") ctx.ModuleBuild(pctx, android.ModuleBuildParams{ Rule: jarjar, Description: "jarjar", Output: outputFile, Input: classesJar, Implicit: rulesFile, Args: map[string]string{ "rulesFile": rulesFile.String(), }, }) return outputFile }