// Copyright (C) 2021 The Android Open Source Project // // 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 apex import ( "android/soong/android" ) // Contains 'deapexer' a private module type used by 'prebuilt_apex' to make dex files contained // within a .apex file referenced by `prebuilt_apex` available for use by their associated // `java_import` modules. // // An 'apex' module references `java_library` modules from which .dex files are obtained that are // stored in the resulting `.apex` file. The resulting `.apex` file is then made available as a // prebuilt by referencing it from a `prebuilt_apex`. For each such `java_library` that is used by // modules outside the `.apex` file a `java_import` prebuilt is made available referencing a jar // that contains the Java classes. // // When building a Java module type, e.g. `java_module` or `android_app` against such prebuilts the // `java_import` provides the classes jar (jar containing `.class` files) against which the // module's `.java` files are compiled. That classes jar usually contains only stub classes. The // resulting classes jar is converted into a dex jar (jar containing `.dex` files). Then if // necessary the dex jar is further processed by `dexpreopt` to produce an optimized form of the // library specific to the current Android version. This process requires access to implementation // dex jars for each `java_import`. The `java_import` will obtain the implementation dex jar from // the `.apex` file in the associated `prebuilt_apex`. // // This is intentionally not registered by name as it is not intended to be used from within an // `Android.bp` file. // DeapexerExportedFile defines the properties needed to expose a file from the deapexer module. type DeapexerExportedFile struct { // The tag parameter which must be passed to android.OutputFileProducer OutputFiles(tag) method // to retrieve the path to the unpacked file. Tag string // The path within the APEX that needs to be exported. Path string `android:"path"` } // DeapexerProperties specifies the properties supported by the deapexer module. // // As these are never intended to be supplied in a .bp file they use a different naming convention // to make it clear that they are different. type DeapexerProperties struct { // List of common modules that may need access to files exported by this module. // // A common module in this sense is one that is not arch specific but uses a common variant for // all architectures, e.g. java. CommonModules []string // List of files exported from the .apex file by this module ExportedFiles []DeapexerExportedFile } type SelectedApexProperties struct { // The path to the apex selected for use by this module. // // Is tagged as `android:"path"` because it will usually contain a string of the form ":" // and is tagged as "`blueprint:"mutate"` because it is only initialized in a LoadHook not an // Android.bp file. Selected_apex *string `android:"path" blueprint:"mutated"` } type Deapexer struct { android.ModuleBase properties DeapexerProperties selectedApexProperties SelectedApexProperties inputApex android.Path } func privateDeapexerFactory() android.Module { module := &Deapexer{} module.AddProperties(&module.properties, &module.selectedApexProperties) android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon) return module } func (p *Deapexer) DepsMutator(ctx android.BottomUpMutatorContext) { // Add dependencies from the java modules to which this exports files from the `.apex` file onto // this module so that they can access the `DeapexerInfo` object that this provides. for _, lib := range p.properties.CommonModules { dep := prebuiltApexExportedModuleName(ctx, lib) ctx.AddReverseDependency(ctx.Module(), android.DeapexerTag, dep) } } func (p *Deapexer) GenerateAndroidBuildActions(ctx android.ModuleContext) { p.inputApex = android.OptionalPathForModuleSrc(ctx, p.selectedApexProperties.Selected_apex).Path() // Create and remember the directory into which the .apex file's contents will be unpacked. deapexerOutput := android.PathForModuleOut(ctx, "deapexer") exports := make(map[string]android.Path) // Create mappings from name+tag to all the required exported paths. for _, e := range p.properties.ExportedFiles { tag := e.Tag path := e.Path // Populate the exports that this makes available. exports[tag] = deapexerOutput.Join(ctx, path) } // If the prebuilt_apex exports any files then create a build rule that unpacks the apex using // deapexer and verifies that all the required files were created. Also, make the mapping from // name+tag to path available for other modules. if len(exports) > 0 { // Make the information available for other modules. ctx.SetProvider(android.DeapexerProvider, android.NewDeapexerInfo(exports)) // Create a sorted list of the files that this exports. exportedPaths := make(android.Paths, 0, len(exports)) for _, p := range exports { exportedPaths = append(exportedPaths, p) } exportedPaths = android.SortedUniquePaths(exportedPaths) // The apex needs to export some files so create a ninja rule to unpack the apex and check that // the required files are present. builder := android.NewRuleBuilder(pctx, ctx) command := builder.Command() command. Tool(android.PathForSource(ctx, "build/soong/scripts/unpack-prebuilt-apex.sh")). BuiltTool("deapexer"). BuiltTool("debugfs"). Input(p.inputApex). Text(deapexerOutput.String()) for _, p := range exportedPaths { command.Output(p.(android.WritablePath)) } builder.Build("deapexer", "deapex "+ctx.ModuleName()) } }