blob: a6731080c2ddedd95e03ce30db1c5a1a86c3223d [file] [log] [blame]
Paul Duffin064b70c2020-11-02 17:32:38 +00001// Copyright (C) 2021 The Android Open Source Project
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package apex
16
17import (
Martin Stjernholm44825602021-09-17 01:44:12 +010018 "strings"
19
Paul Duffin064b70c2020-11-02 17:32:38 +000020 "android/soong/android"
21)
22
23// Contains 'deapexer' a private module type used by 'prebuilt_apex' to make dex files contained
24// within a .apex file referenced by `prebuilt_apex` available for use by their associated
25// `java_import` modules.
26//
27// An 'apex' module references `java_library` modules from which .dex files are obtained that are
28// stored in the resulting `.apex` file. The resulting `.apex` file is then made available as a
29// prebuilt by referencing it from a `prebuilt_apex`. For each such `java_library` that is used by
30// modules outside the `.apex` file a `java_import` prebuilt is made available referencing a jar
31// that contains the Java classes.
32//
33// When building a Java module type, e.g. `java_module` or `android_app` against such prebuilts the
34// `java_import` provides the classes jar (jar containing `.class` files) against which the
35// module's `.java` files are compiled. That classes jar usually contains only stub classes. The
36// resulting classes jar is converted into a dex jar (jar containing `.dex` files). Then if
37// necessary the dex jar is further processed by `dexpreopt` to produce an optimized form of the
38// library specific to the current Android version. This process requires access to implementation
39// dex jars for each `java_import`. The `java_import` will obtain the implementation dex jar from
40// the `.apex` file in the associated `prebuilt_apex`.
41//
42// This is intentionally not registered by name as it is not intended to be used from within an
43// `Android.bp` file.
44
Paul Duffin3bae0682021-05-05 18:03:47 +010045// DeapexerProperties specifies the properties supported by the deapexer module.
46//
47// As these are never intended to be supplied in a .bp file they use a different naming convention
48// to make it clear that they are different.
49type DeapexerProperties struct {
50 // List of common modules that may need access to files exported by this module.
51 //
52 // A common module in this sense is one that is not arch specific but uses a common variant for
53 // all architectures, e.g. java.
54 CommonModules []string
55
Spandan Das2ea84dd2024-01-25 22:12:50 +000056 // List of modules that use an embedded .prof to guide optimization of the equivalent dexpreopt artifact
57 // This is a subset of CommonModules
58 DexpreoptProfileGuidedModules []string
59
Paul Duffin3bae0682021-05-05 18:03:47 +010060 // List of files exported from the .apex file by this module
Paul Duffinb4bbf2c2021-06-17 15:59:07 +010061 //
62 // Each entry is a path from the apex root, e.g. javalib/core-libart.jar.
63 ExportedFiles []string
Paul Duffin064b70c2020-11-02 17:32:38 +000064}
65
Paul Duffin11216db2021-03-01 14:14:52 +000066type SelectedApexProperties struct {
67 // The path to the apex selected for use by this module.
68 //
69 // Is tagged as `android:"path"` because it will usually contain a string of the form ":<module>"
70 // and is tagged as "`blueprint:"mutate"` because it is only initialized in a LoadHook not an
71 // Android.bp file.
72 Selected_apex *string `android:"path" blueprint:"mutated"`
73}
74
Paul Duffin064b70c2020-11-02 17:32:38 +000075type Deapexer struct {
76 android.ModuleBase
Paul Duffin064b70c2020-11-02 17:32:38 +000077
Paul Duffin11216db2021-03-01 14:14:52 +000078 properties DeapexerProperties
79 selectedApexProperties SelectedApexProperties
Paul Duffin064b70c2020-11-02 17:32:38 +000080
81 inputApex android.Path
82}
83
Martin Stjernholm44825602021-09-17 01:44:12 +010084// Returns the name of the deapexer module corresponding to an APEX module with the given name.
85func deapexerModuleName(apexModuleName string) string {
86 return apexModuleName + ".deapexer"
87}
88
89// Returns the name of the APEX module corresponding to an deapexer module with
90// the given name. This reverses deapexerModuleName.
91func apexModuleName(deapexerModuleName string) string {
92 return strings.TrimSuffix(deapexerModuleName, ".deapexer")
93}
94
Paul Duffin064b70c2020-11-02 17:32:38 +000095func privateDeapexerFactory() android.Module {
96 module := &Deapexer{}
Paul Duffin11216db2021-03-01 14:14:52 +000097 module.AddProperties(&module.properties, &module.selectedApexProperties)
Paul Duffin064b70c2020-11-02 17:32:38 +000098 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon)
99 return module
100}
101
Liz Kammer356f7d42021-01-26 09:18:53 -0500102func (p *Deapexer) DepsMutator(ctx android.BottomUpMutatorContext) {
Paul Duffin064b70c2020-11-02 17:32:38 +0000103 // Add dependencies from the java modules to which this exports files from the `.apex` file onto
104 // this module so that they can access the `DeapexerInfo` object that this provides.
Spandan Das5be63332023-12-13 00:06:32 +0000105 // TODO: b/308174306 - Once all the mainline modules have been flagged, drop this dependency edge
Paul Duffin3bae0682021-05-05 18:03:47 +0100106 for _, lib := range p.properties.CommonModules {
Paul Duffin064b70c2020-11-02 17:32:38 +0000107 dep := prebuiltApexExportedModuleName(ctx, lib)
108 ctx.AddReverseDependency(ctx.Module(), android.DeapexerTag, dep)
109 }
110}
111
112func (p *Deapexer) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Paul Duffin11216db2021-03-01 14:14:52 +0000113 p.inputApex = android.OptionalPathForModuleSrc(ctx, p.selectedApexProperties.Selected_apex).Path()
Paul Duffin064b70c2020-11-02 17:32:38 +0000114
115 // Create and remember the directory into which the .apex file's contents will be unpacked.
116 deapexerOutput := android.PathForModuleOut(ctx, "deapexer")
117
Jiakai Zhang204356f2021-09-09 08:12:46 +0000118 exports := make(map[string]android.WritablePath)
Paul Duffin064b70c2020-11-02 17:32:38 +0000119
Paul Duffinb4bbf2c2021-06-17 15:59:07 +0100120 // Create mappings from apex relative path to the extracted file's path.
121 exportedPaths := make(android.Paths, 0, len(exports))
122 for _, path := range p.properties.ExportedFiles {
Paul Duffin3bae0682021-05-05 18:03:47 +0100123 // Populate the exports that this makes available.
Paul Duffinb4bbf2c2021-06-17 15:59:07 +0100124 extractedPath := deapexerOutput.Join(ctx, path)
125 exports[path] = extractedPath
126 exportedPaths = append(exportedPaths, extractedPath)
Paul Duffin064b70c2020-11-02 17:32:38 +0000127 }
128
129 // If the prebuilt_apex exports any files then create a build rule that unpacks the apex using
130 // deapexer and verifies that all the required files were created. Also, make the mapping from
Paul Duffinb4bbf2c2021-06-17 15:59:07 +0100131 // apex relative path to extracted file path available for other modules.
Paul Duffin064b70c2020-11-02 17:32:38 +0000132 if len(exports) > 0 {
133 // Make the information available for other modules.
Spandan Das5be63332023-12-13 00:06:32 +0000134 di := android.NewDeapexerInfo(apexModuleName(ctx.ModuleName()), exports, p.properties.CommonModules)
Spandan Das2ea84dd2024-01-25 22:12:50 +0000135 di.AddDexpreoptProfileGuidedExportedModuleNames(p.properties.DexpreoptProfileGuidedModules...)
Colin Cross40213022023-12-13 15:19:49 -0800136 android.SetProvider(ctx, android.DeapexerProvider, di)
Paul Duffin064b70c2020-11-02 17:32:38 +0000137
138 // Create a sorted list of the files that this exports.
Paul Duffin064b70c2020-11-02 17:32:38 +0000139 exportedPaths = android.SortedUniquePaths(exportedPaths)
140
141 // The apex needs to export some files so create a ninja rule to unpack the apex and check that
142 // the required files are present.
143 builder := android.NewRuleBuilder(pctx, ctx)
144 command := builder.Command()
145 command.
146 Tool(android.PathForSource(ctx, "build/soong/scripts/unpack-prebuilt-apex.sh")).
147 BuiltTool("deapexer").
148 BuiltTool("debugfs").
Paul Duffine17c3162022-12-12 17:37:20 +0000149 BuiltTool("fsck.erofs").
Paul Duffin064b70c2020-11-02 17:32:38 +0000150 Input(p.inputApex).
151 Text(deapexerOutput.String())
152 for _, p := range exportedPaths {
153 command.Output(p.(android.WritablePath))
154 }
Martin Stjernholm44825602021-09-17 01:44:12 +0100155 builder.Build("deapexer", "deapex "+apexModuleName(ctx.ModuleName()))
Paul Duffin064b70c2020-11-02 17:32:38 +0000156 }
157}