3bae068ee5
Previously, the deapexer module had to duplicate the java library specific logic for constructing the path to the library's dex file in the .apex file. That is not something that the deapexer needs to be aware of, all it needs is a list of files that should be exported. This change moves that logic into the prebuilt_apex/apex_set modules and generalizes the deapexer module so that it can export any files that are requested. The deapexer module does still need to know which java modules need access to exported files so it can add dependencies from them onto itself. However, it does not need to know what the type of the module is. Bug: 186455808 Test: m nothing m SOONG_CONFIG_art_module_source_build=false nothing Change-Id: I71c6f0f761efe3b6d66d54273786e98cd545811c
148 lines
6.1 KiB
Go
148 lines
6.1 KiB
Go
// 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 ":<module>"
|
|
// 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())
|
|
}
|
|
}
|