// 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" "strings" "github.com/google/blueprint" "github.com/google/blueprint/proptools" "android/soong/android" "android/soong/remoteexec" ) var ( Signapk, SignapkRE = pctx.RemoteStaticRules("signapk", blueprint.RuleParams{ Command: `rm -f $out && $reTemplate${config.JavaCmd} ${config.JavaVmFlags} -Djava.library.path=$$(dirname ${config.SignapkJniLibrary}) ` + `-jar ${config.SignapkCmd} $flags $certificates $in $out`, CommandDeps: []string{"${config.SignapkCmd}", "${config.SignapkJniLibrary}"}, }, &remoteexec.REParams{Labels: map[string]string{"type": "tool", "name": "signapk"}, ExecStrategy: "${config.RESignApkExecStrategy}", Inputs: []string{"${config.SignapkCmd}", "$in", "$$(dirname ${config.SignapkJniLibrary})", "$implicits"}, OutputFiles: []string{"$outCommaList"}, ToolchainInputs: []string{"${config.JavaCmd}"}, Platform: map[string]string{remoteexec.PoolKey: "${config.REJavaPool}"}, }, []string{"flags", "certificates"}, []string{"implicits", "outCommaList"}) ) var combineApk = pctx.AndroidStaticRule("combineApk", blueprint.RuleParams{ Command: `${config.MergeZipsCmd} $out $in`, CommandDeps: []string{"${config.MergeZipsCmd}"}, }) func CreateAndSignAppPackage(ctx android.ModuleContext, outputFile android.WritablePath, packageFile, jniJarFile, dexJarFile android.Path, certificates []Certificate, deps android.Paths, v4SignatureFile android.WritablePath, lineageFile android.Path, rotationMinSdkVersion string, shrinkResources bool) { unsignedApkName := strings.TrimSuffix(outputFile.Base(), ".apk") + "-unsigned.apk" unsignedApk := android.PathForModuleOut(ctx, unsignedApkName) var inputs android.Paths if dexJarFile != nil { inputs = append(inputs, dexJarFile) } inputs = append(inputs, packageFile) if jniJarFile != nil { inputs = append(inputs, jniJarFile) } ctx.Build(pctx, android.BuildParams{ Rule: combineApk, Inputs: inputs, Output: unsignedApk, Implicits: deps, }) if shrinkResources { shrunkenApk := android.PathForModuleOut(ctx, "resource-shrunken", unsignedApk.Base()) ShrinkResources(ctx, unsignedApk, shrunkenApk) unsignedApk = shrunkenApk } SignAppPackage(ctx, outputFile, unsignedApk, certificates, v4SignatureFile, lineageFile, rotationMinSdkVersion) } func SignAppPackage(ctx android.ModuleContext, signedApk android.WritablePath, unsignedApk android.Path, certificates []Certificate, v4SignatureFile android.WritablePath, lineageFile android.Path, rotationMinSdkVersion string) { var certificateArgs []string var deps android.Paths for _, c := range certificates { certificateArgs = append(certificateArgs, c.Pem.String(), c.Key.String()) deps = append(deps, c.Pem, c.Key) } outputFiles := android.WritablePaths{signedApk} var flags []string if v4SignatureFile != nil { outputFiles = append(outputFiles, v4SignatureFile) flags = append(flags, "--enable-v4") } if lineageFile != nil { flags = append(flags, "--lineage", lineageFile.String()) deps = append(deps, lineageFile) } if rotationMinSdkVersion != "" { flags = append(flags, "--rotation-min-sdk-version", rotationMinSdkVersion) } rule := Signapk args := map[string]string{ "certificates": strings.Join(certificateArgs, " "), "flags": strings.Join(flags, " "), } if ctx.Config().UseRBE() && ctx.Config().IsEnvTrue("RBE_SIGNAPK") { rule = SignapkRE args["implicits"] = strings.Join(deps.Strings(), ",") args["outCommaList"] = strings.Join(outputFiles.Strings(), ",") } ctx.Build(pctx, android.BuildParams{ Rule: rule, Description: "signapk", Outputs: outputFiles, Input: unsignedApk, Implicits: deps, Args: args, }) } var buildAAR = pctx.AndroidStaticRule("buildAAR", blueprint.RuleParams{ Command: `rm -rf ${outDir} && mkdir -p ${outDir} && ` + `cp ${manifest} ${outDir}/AndroidManifest.xml && ` + `cp ${classesJar} ${outDir}/classes.jar && ` + `cp ${rTxt} ${outDir}/R.txt && ` + `${config.SoongZipCmd} -jar -o $out -C ${outDir} -D ${outDir}`, CommandDeps: []string{"${config.SoongZipCmd}"}, }, "manifest", "classesJar", "rTxt", "outDir") func BuildAAR(ctx android.ModuleContext, outputFile android.WritablePath, classesJar, manifest, rTxt android.Path, res android.Paths) { // TODO(ccross): uniquify and copy resources with dependencies deps := android.Paths{manifest, rTxt} classesJarPath := "" if classesJar != nil { deps = append(deps, classesJar) classesJarPath = classesJar.String() } ctx.Build(pctx, android.BuildParams{ Rule: buildAAR, Description: "aar", Implicits: deps, Output: outputFile, Args: map[string]string{ "manifest": manifest.String(), "classesJar": classesJarPath, "rTxt": rTxt.String(), "outDir": android.PathForModuleOut(ctx, "aar").String(), }, }) } var buildBundleModule = pctx.AndroidStaticRule("buildBundleModule", blueprint.RuleParams{ Command: `${config.MergeZipsCmd} ${out} ${in}`, CommandDeps: []string{"${config.MergeZipsCmd}"}, }) var bundleMungePackage = pctx.AndroidStaticRule("bundleMungePackage", blueprint.RuleParams{ Command: `${config.Zip2ZipCmd} -i ${in} -o ${out} AndroidManifest.xml:manifest/AndroidManifest.xml resources.pb "res/**/*" "assets/**/*"`, CommandDeps: []string{"${config.Zip2ZipCmd}"}, }) var bundleMungeDexJar = pctx.AndroidStaticRule("bundleMungeDexJar", blueprint.RuleParams{ Command: `${config.Zip2ZipCmd} -i ${in} -o ${out} "classes*.dex:dex/" && ` + `${config.Zip2ZipCmd} -i ${in} -o ${resJar} -x "classes*.dex" "**/*:root/"`, CommandDeps: []string{"${config.Zip2ZipCmd}"}, }, "resJar") // Builds an app into a module suitable for input to bundletool func BuildBundleModule(ctx android.ModuleContext, outputFile android.WritablePath, packageFile, jniJarFile, dexJarFile android.Path) { protoResJarFile := android.PathForModuleOut(ctx, "package-res.pb.apk") aapt2Convert(ctx, protoResJarFile, packageFile, "proto") var zips android.Paths mungedPackage := android.PathForModuleOut(ctx, "bundle", "apk.zip") ctx.Build(pctx, android.BuildParams{ Rule: bundleMungePackage, Input: protoResJarFile, Output: mungedPackage, Description: "bundle apk", }) zips = append(zips, mungedPackage) if dexJarFile != nil { mungedDexJar := android.PathForModuleOut(ctx, "bundle", "dex.zip") mungedResJar := android.PathForModuleOut(ctx, "bundle", "res.zip") ctx.Build(pctx, android.BuildParams{ Rule: bundleMungeDexJar, Input: dexJarFile, Output: mungedDexJar, ImplicitOutput: mungedResJar, Description: "bundle dex", Args: map[string]string{ "resJar": mungedResJar.String(), }, }) zips = append(zips, mungedDexJar, mungedResJar) } if jniJarFile != nil { zips = append(zips, jniJarFile) } ctx.Build(pctx, android.BuildParams{ Rule: buildBundleModule, Inputs: zips, Output: outputFile, Description: "bundle", }) } func TransformJniLibsToJar( ctx android.ModuleContext, outputFile android.WritablePath, jniLibs []jniLib, prebuiltJniPackages android.Paths, uncompressJNI bool) { var deps android.Paths jarArgs := []string{ "-j", // junk paths, they will be added back with -P arguments } if uncompressJNI { jarArgs = append(jarArgs, "-L", "0") } for _, j := range jniLibs { deps = append(deps, j.path) jarArgs = append(jarArgs, "-P", targetToJniDir(j.target), "-f", j.path.String()) } rule := zip args := map[string]string{ "jarArgs": strings.Join(proptools.NinjaAndShellEscapeList(jarArgs), " "), } if ctx.Config().UseRBE() && ctx.Config().IsEnvTrue("RBE_ZIP") { rule = zipRE args["implicits"] = strings.Join(deps.Strings(), ",") } var jniJarPath android.WritablePath = android.PathForModuleOut(ctx, "jniJarOutput.zip") if len(prebuiltJniPackages) == 0 { jniJarPath = outputFile } ctx.Build(pctx, android.BuildParams{ Rule: rule, Description: "zip jni libs", Output: jniJarPath, Implicits: deps, Args: args, }) if len(prebuiltJniPackages) > 0 { var mergeJniJarPath android.WritablePath = android.PathForModuleOut(ctx, "mergeJniJarOutput.zip") if !uncompressJNI { mergeJniJarPath = outputFile } ctx.Build(pctx, android.BuildParams{ Rule: mergeAssetsRule, Description: "merge prebuilt JNI packages", Inputs: append(prebuiltJniPackages, jniJarPath), Output: mergeJniJarPath, }) if uncompressJNI { ctx.Build(pctx, android.BuildParams{ Rule: uncompressEmbeddedJniLibsRule, Input: mergeJniJarPath, Output: outputFile, }) } } } func (a *AndroidApp) generateJavaUsedByApex(ctx android.ModuleContext) { javaApiUsedByOutputFile := android.PathForModuleOut(ctx, a.installApkName+"_using.xml") javaUsedByRule := android.NewRuleBuilder(pctx, ctx) javaUsedByRule.Command(). Tool(android.PathForSource(ctx, "build/soong/scripts/gen_java_usedby_apex.sh")). BuiltTool("dexdeps"). Output(javaApiUsedByOutputFile). Input(a.Library.Module.outputFile) javaUsedByRule.Build("java_usedby_list", "Generate Java APIs used by Apex") a.javaApiUsedByOutputFile = javaApiUsedByOutputFile } func targetToJniDir(target android.Target) string { return filepath.Join("lib", target.Arch.Abi[0]) }