Paul Duffin | bb7f1ac | 2021-03-29 22:18:45 +0100 | [diff] [blame] | 1 | // Copyright 2021 Google Inc. All rights reserved. |
| 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 | |
| 15 | package java |
| 16 | |
| 17 | import ( |
Paul Duffin | 6a76645 | 2021-04-12 14:15:22 +0100 | [diff] [blame] | 18 | "fmt" |
| 19 | |
Paul Duffin | bb7f1ac | 2021-03-29 22:18:45 +0100 | [diff] [blame] | 20 | "android/soong/android" |
| 21 | "android/soong/dexpreopt" |
| 22 | ) |
| 23 | |
| 24 | func init() { |
| 25 | registerPlatformBootclasspathBuildComponents(android.InitRegistrationContext) |
| 26 | } |
| 27 | |
| 28 | func registerPlatformBootclasspathBuildComponents(ctx android.RegistrationContext) { |
LaMont Jones | 0c10e4d | 2023-05-16 00:58:37 +0000 | [diff] [blame^] | 29 | ctx.RegisterParallelSingletonModuleType("platform_bootclasspath", platformBootclasspathFactory) |
Paul Duffin | bb7f1ac | 2021-03-29 22:18:45 +0100 | [diff] [blame] | 30 | } |
| 31 | |
Paul Duffin | 01b463b | 2021-04-26 20:05:39 +0100 | [diff] [blame] | 32 | // The tags used for the dependencies between the platform bootclasspath and any configured boot |
| 33 | // jars. |
| 34 | var ( |
satayev | d604b21 | 2021-07-21 14:23:52 +0100 | [diff] [blame] | 35 | platformBootclasspathArtBootJarDepTag = bootclasspathDependencyTag{name: "art-boot-jar"} |
| 36 | platformBootclasspathBootJarDepTag = bootclasspathDependencyTag{name: "platform-boot-jar"} |
| 37 | platformBootclasspathApexBootJarDepTag = bootclasspathDependencyTag{name: "apex-boot-jar"} |
Paul Duffin | 01b463b | 2021-04-26 20:05:39 +0100 | [diff] [blame] | 38 | ) |
Paul Duffin | b432df9 | 2021-03-22 22:09:42 +0000 | [diff] [blame] | 39 | |
Paul Duffin | bb7f1ac | 2021-03-29 22:18:45 +0100 | [diff] [blame] | 40 | type platformBootclasspathModule struct { |
Paul Duffin | e3ecce6 | 2021-04-29 10:34:11 +0100 | [diff] [blame] | 41 | android.SingletonModuleBase |
Artur Satayev | 97259dc | 2021-04-07 15:17:14 +0100 | [diff] [blame] | 42 | ClasspathFragmentBase |
Paul Duffin | b432df9 | 2021-03-22 22:09:42 +0000 | [diff] [blame] | 43 | |
Paul Duffin | 62d8c3b | 2021-04-07 20:35:11 +0100 | [diff] [blame] | 44 | properties platformBootclasspathProperties |
| 45 | |
Paul Duffin | b432df9 | 2021-03-22 22:09:42 +0000 | [diff] [blame] | 46 | // The apex:module pairs obtained from the configured modules. |
Paul Duffin | b432df9 | 2021-03-22 22:09:42 +0000 | [diff] [blame] | 47 | configuredModules []android.Module |
Paul Duffin | 62d8c3b | 2021-04-07 20:35:11 +0100 | [diff] [blame] | 48 | |
| 49 | // The apex:module pairs obtained from the fragments. |
Paul Duffin | 62d8c3b | 2021-04-07 20:35:11 +0100 | [diff] [blame] | 50 | fragments []android.Module |
Paul Duffin | 6a76645 | 2021-04-12 14:15:22 +0100 | [diff] [blame] | 51 | |
| 52 | // Path to the monolithic hiddenapi-flags.csv file. |
Paul Duffin | 0b65986 | 2021-04-13 13:02:29 +0100 | [diff] [blame] | 53 | hiddenAPIFlagsCSV android.OutputPath |
Paul Duffin | 6a76645 | 2021-04-12 14:15:22 +0100 | [diff] [blame] | 54 | |
| 55 | // Path to the monolithic hiddenapi-index.csv file. |
Paul Duffin | 0b65986 | 2021-04-13 13:02:29 +0100 | [diff] [blame] | 56 | hiddenAPIIndexCSV android.OutputPath |
Paul Duffin | 6a76645 | 2021-04-12 14:15:22 +0100 | [diff] [blame] | 57 | |
| 58 | // Path to the monolithic hiddenapi-unsupported.csv file. |
Paul Duffin | 0b65986 | 2021-04-13 13:02:29 +0100 | [diff] [blame] | 59 | hiddenAPIMetadataCSV android.OutputPath |
Paul Duffin | 62d8c3b | 2021-04-07 20:35:11 +0100 | [diff] [blame] | 60 | } |
| 61 | |
Paul Duffin | 62d8c3b | 2021-04-07 20:35:11 +0100 | [diff] [blame] | 62 | type platformBootclasspathProperties struct { |
Paul Duffin | b67d878 | 2021-04-22 11:49:41 +0100 | [diff] [blame] | 63 | BootclasspathFragmentsDepsProperties |
Paul Duffin | 702210b | 2021-04-08 20:12:41 +0100 | [diff] [blame] | 64 | |
Paul Duffin | 9b61abb | 2022-07-27 16:16:54 +0000 | [diff] [blame] | 65 | HiddenAPIFlagFileProperties |
Paul Duffin | bb7f1ac | 2021-03-29 22:18:45 +0100 | [diff] [blame] | 66 | } |
| 67 | |
Paul Duffin | e3ecce6 | 2021-04-29 10:34:11 +0100 | [diff] [blame] | 68 | func platformBootclasspathFactory() android.SingletonModule { |
Paul Duffin | bb7f1ac | 2021-03-29 22:18:45 +0100 | [diff] [blame] | 69 | m := &platformBootclasspathModule{} |
Paul Duffin | 62d8c3b | 2021-04-07 20:35:11 +0100 | [diff] [blame] | 70 | m.AddProperties(&m.properties) |
satayev | 95e9c5b | 2021-04-29 11:50:26 +0100 | [diff] [blame] | 71 | initClasspathFragment(m, BOOTCLASSPATH) |
Paul Duffin | bb7f1ac | 2021-03-29 22:18:45 +0100 | [diff] [blame] | 72 | android.InitAndroidArchModule(m, android.DeviceSupported, android.MultilibCommon) |
| 73 | return m |
| 74 | } |
| 75 | |
Paul Duffin | 6a76645 | 2021-04-12 14:15:22 +0100 | [diff] [blame] | 76 | var _ android.OutputFileProducer = (*platformBootclasspathModule)(nil) |
| 77 | |
Artur Satayev | 97259dc | 2021-04-07 15:17:14 +0100 | [diff] [blame] | 78 | func (b *platformBootclasspathModule) AndroidMkEntries() (entries []android.AndroidMkEntries) { |
| 79 | entries = append(entries, android.AndroidMkEntries{ |
| 80 | Class: "FAKE", |
| 81 | // Need at least one output file in order for this to take effect. |
| 82 | OutputFile: android.OptionalPathForPath(b.hiddenAPIFlagsCSV), |
| 83 | Include: "$(BUILD_PHONY_PACKAGE)", |
| 84 | }) |
satayev | 128ce2f | 2021-05-06 13:21:15 +0100 | [diff] [blame] | 85 | entries = append(entries, b.classpathFragmentBase().androidMkEntries()...) |
Artur Satayev | 97259dc | 2021-04-07 15:17:14 +0100 | [diff] [blame] | 86 | return |
Paul Duffin | 6a76645 | 2021-04-12 14:15:22 +0100 | [diff] [blame] | 87 | } |
| 88 | |
| 89 | // Make the hidden API files available from the platform-bootclasspath module. |
| 90 | func (b *platformBootclasspathModule) OutputFiles(tag string) (android.Paths, error) { |
| 91 | switch tag { |
| 92 | case "hiddenapi-flags.csv": |
| 93 | return android.Paths{b.hiddenAPIFlagsCSV}, nil |
| 94 | case "hiddenapi-index.csv": |
| 95 | return android.Paths{b.hiddenAPIIndexCSV}, nil |
| 96 | case "hiddenapi-metadata.csv": |
| 97 | return android.Paths{b.hiddenAPIMetadataCSV}, nil |
| 98 | } |
| 99 | |
| 100 | return nil, fmt.Errorf("unknown tag %s", tag) |
| 101 | } |
| 102 | |
Paul Duffin | bb7f1ac | 2021-03-29 22:18:45 +0100 | [diff] [blame] | 103 | func (b *platformBootclasspathModule) DepsMutator(ctx android.BottomUpMutatorContext) { |
Paul Duffin | 74431d5 | 2021-04-21 14:10:42 +0100 | [diff] [blame] | 104 | b.hiddenAPIDepsMutator(ctx) |
| 105 | |
Jiakai Zhang | bc698cd | 2023-05-08 16:28:38 +0000 | [diff] [blame] | 106 | if !dexpreopt.IsDex2oatNeeded(ctx) { |
Qiao Yang | 8d8c660 | 2023-05-05 15:03:24 +0000 | [diff] [blame] | 107 | return |
| 108 | } |
| 109 | |
Paul Duffin | bb7f1ac | 2021-03-29 22:18:45 +0100 | [diff] [blame] | 110 | // Add a dependency onto the dex2oat tool which is needed for creating the boot image. The |
| 111 | // path is retrieved from the dependency by GetGlobalSoongConfig(ctx). |
| 112 | dexpreopt.RegisterToolDeps(ctx) |
| 113 | } |
| 114 | |
Paul Duffin | 74431d5 | 2021-04-21 14:10:42 +0100 | [diff] [blame] | 115 | func (b *platformBootclasspathModule) hiddenAPIDepsMutator(ctx android.BottomUpMutatorContext) { |
| 116 | if ctx.Config().IsEnvTrue("UNSAFE_DISABLE_HIDDENAPI_FLAGS") { |
| 117 | return |
| 118 | } |
| 119 | |
| 120 | // Add dependencies onto the stub lib modules. |
Paul Duffin | 31fad80 | 2021-06-18 18:14:25 +0100 | [diff] [blame] | 121 | apiLevelToStubLibModules := hiddenAPIComputeMonolithicStubLibModules(ctx.Config()) |
| 122 | hiddenAPIAddStubLibDependencies(ctx, apiLevelToStubLibModules) |
Paul Duffin | 74431d5 | 2021-04-21 14:10:42 +0100 | [diff] [blame] | 123 | } |
| 124 | |
Paul Duffin | 4994d26 | 2021-04-22 12:08:59 +0100 | [diff] [blame] | 125 | func (b *platformBootclasspathModule) BootclasspathDepsMutator(ctx android.BottomUpMutatorContext) { |
| 126 | // Add dependencies on all the modules configured in the "art" boot image. |
| 127 | artImageConfig := genBootImageConfigs(ctx)[artBootImageName] |
Paul Duffin | 01b463b | 2021-04-26 20:05:39 +0100 | [diff] [blame] | 128 | addDependenciesOntoBootImageModules(ctx, artImageConfig.modules, platformBootclasspathArtBootJarDepTag) |
Paul Duffin | b432df9 | 2021-03-22 22:09:42 +0000 | [diff] [blame] | 129 | |
Paul Duffin | 01b463b | 2021-04-26 20:05:39 +0100 | [diff] [blame] | 130 | // Add dependencies on all the non-updatable module configured in the "boot" boot image. That does |
| 131 | // not include modules configured in the "art" boot image. |
Jiakai Zhang | c08c162 | 2023-05-10 18:38:34 +0100 | [diff] [blame] | 132 | addDependenciesOntoBootImageModules(ctx, b.platformJars(ctx), platformBootclasspathBootJarDepTag) |
Paul Duffin | b432df9 | 2021-03-22 22:09:42 +0000 | [diff] [blame] | 133 | |
satayev | d604b21 | 2021-07-21 14:23:52 +0100 | [diff] [blame] | 134 | // Add dependencies on all the apex jars. |
| 135 | apexJars := dexpreopt.GetGlobalConfig(ctx).ApexBootJars |
| 136 | addDependenciesOntoBootImageModules(ctx, apexJars, platformBootclasspathApexBootJarDepTag) |
Paul Duffin | 62d8c3b | 2021-04-07 20:35:11 +0100 | [diff] [blame] | 137 | |
Paul Duffin | 4994d26 | 2021-04-22 12:08:59 +0100 | [diff] [blame] | 138 | // Add dependencies on all the fragments. |
| 139 | b.properties.BootclasspathFragmentsDepsProperties.addDependenciesOntoFragments(ctx) |
Paul Duffin | b432df9 | 2021-03-22 22:09:42 +0000 | [diff] [blame] | 140 | } |
| 141 | |
Paul Duffin | 01b463b | 2021-04-26 20:05:39 +0100 | [diff] [blame] | 142 | func addDependenciesOntoBootImageModules(ctx android.BottomUpMutatorContext, modules android.ConfiguredJarList, tag bootclasspathDependencyTag) { |
Paul Duffin | b432df9 | 2021-03-22 22:09:42 +0000 | [diff] [blame] | 143 | for i := 0; i < modules.Len(); i++ { |
| 144 | apex := modules.Apex(i) |
| 145 | name := modules.Jar(i) |
| 146 | |
Paul Duffin | 01b463b | 2021-04-26 20:05:39 +0100 | [diff] [blame] | 147 | addDependencyOntoApexModulePair(ctx, apex, name, tag) |
Paul Duffin | b432df9 | 2021-03-22 22:09:42 +0000 | [diff] [blame] | 148 | } |
| 149 | } |
| 150 | |
Paul Duffin | e3ecce6 | 2021-04-29 10:34:11 +0100 | [diff] [blame] | 151 | // GenerateSingletonBuildActions does nothing and must never do anything. |
| 152 | // |
| 153 | // This module only implements android.SingletonModule so that it can implement |
| 154 | // android.SingletonMakeVarsProvider. |
| 155 | func (b *platformBootclasspathModule) GenerateSingletonBuildActions(android.SingletonContext) { |
| 156 | // Keep empty |
| 157 | } |
| 158 | |
| 159 | func (d *platformBootclasspathModule) MakeVars(ctx android.MakeVarsContext) { |
Paul Duffin | 12d29b7 | 2021-04-29 13:50:01 +0100 | [diff] [blame] | 160 | d.generateHiddenApiMakeVars(ctx) |
Paul Duffin | e3ecce6 | 2021-04-29 10:34:11 +0100 | [diff] [blame] | 161 | } |
| 162 | |
Paul Duffin | bb7f1ac | 2021-03-29 22:18:45 +0100 | [diff] [blame] | 163 | func (b *platformBootclasspathModule) GenerateAndroidBuildActions(ctx android.ModuleContext) { |
satayev | d604b21 | 2021-07-21 14:23:52 +0100 | [diff] [blame] | 164 | // Gather all the dependencies from the art, platform, and apex boot jars. |
Paul Duffin | 01b463b | 2021-04-26 20:05:39 +0100 | [diff] [blame] | 165 | artModules := gatherApexModulePairDepsWithTag(ctx, platformBootclasspathArtBootJarDepTag) |
satayev | d604b21 | 2021-07-21 14:23:52 +0100 | [diff] [blame] | 166 | platformModules := gatherApexModulePairDepsWithTag(ctx, platformBootclasspathBootJarDepTag) |
| 167 | apexModules := gatherApexModulePairDepsWithTag(ctx, platformBootclasspathApexBootJarDepTag) |
Paul Duffin | 01b463b | 2021-04-26 20:05:39 +0100 | [diff] [blame] | 168 | |
| 169 | // Concatenate them all, in order as they would appear on the bootclasspath. |
| 170 | var allModules []android.Module |
| 171 | allModules = append(allModules, artModules...) |
satayev | d604b21 | 2021-07-21 14:23:52 +0100 | [diff] [blame] | 172 | allModules = append(allModules, platformModules...) |
| 173 | allModules = append(allModules, apexModules...) |
Paul Duffin | 01b463b | 2021-04-26 20:05:39 +0100 | [diff] [blame] | 174 | b.configuredModules = allModules |
| 175 | |
| 176 | // Gather all the fragments dependencies. |
Paul Duffin | 9bacf56 | 2021-04-28 21:16:02 +0100 | [diff] [blame] | 177 | b.fragments = gatherApexModulePairDepsWithTag(ctx, bootclasspathFragmentDepTag) |
Paul Duffin | b432df9 | 2021-03-22 22:09:42 +0000 | [diff] [blame] | 178 | |
Paul Duffin | f23bc47 | 2021-04-27 12:42:20 +0100 | [diff] [blame] | 179 | // Check the configuration of the boot modules. |
| 180 | // ART modules are checked by the art-bootclasspath-fragment. |
satayev | d604b21 | 2021-07-21 14:23:52 +0100 | [diff] [blame] | 181 | b.checkPlatformModules(ctx, platformModules) |
| 182 | b.checkApexModules(ctx, apexModules) |
Paul Duffin | f23bc47 | 2021-04-27 12:42:20 +0100 | [diff] [blame] | 183 | |
satayev | 013485b | 2021-05-06 23:38:10 +0100 | [diff] [blame] | 184 | b.generateClasspathProtoBuildActions(ctx) |
| 185 | |
Paul Duffin | c8ead41 | 2021-06-07 19:28:15 +0100 | [diff] [blame] | 186 | bootDexJarByModule := b.generateHiddenAPIBuildActions(ctx, b.configuredModules, b.fragments) |
| 187 | buildRuleForBootJarsPackageCheck(ctx, bootDexJarByModule) |
Paul Duffin | 702210b | 2021-04-08 20:12:41 +0100 | [diff] [blame] | 188 | |
Jiakai Zhang | c08c162 | 2023-05-10 18:38:34 +0100 | [diff] [blame] | 189 | b.generateBootImageBuildActions(ctx) |
| 190 | b.copyApexBootJarsForAppsDexpreopt(ctx, apexModules) |
Paul Duffin | bb7f1ac | 2021-03-29 22:18:45 +0100 | [diff] [blame] | 191 | } |
| 192 | |
satayev | 013485b | 2021-05-06 23:38:10 +0100 | [diff] [blame] | 193 | // Generate classpaths.proto config |
| 194 | func (b *platformBootclasspathModule) generateClasspathProtoBuildActions(ctx android.ModuleContext) { |
satayev | b309050 | 2021-06-15 17:49:10 +0100 | [diff] [blame] | 195 | configuredJars := b.configuredJars(ctx) |
satayev | 013485b | 2021-05-06 23:38:10 +0100 | [diff] [blame] | 196 | // ART and platform boot jars must have a corresponding entry in DEX2OATBOOTCLASSPATH |
satayev | b309050 | 2021-06-15 17:49:10 +0100 | [diff] [blame] | 197 | classpathJars := configuredJarListToClasspathJars(ctx, configuredJars, BOOTCLASSPATH, DEX2OATBOOTCLASSPATH) |
| 198 | b.classpathFragmentBase().generateClasspathProtoBuildActions(ctx, configuredJars, classpathJars) |
satayev | 013485b | 2021-05-06 23:38:10 +0100 | [diff] [blame] | 199 | } |
| 200 | |
satayev | 142ed27 | 2021-06-15 16:21:17 +0100 | [diff] [blame] | 201 | func (b *platformBootclasspathModule) configuredJars(ctx android.ModuleContext) android.ConfiguredJarList { |
satayev | b309050 | 2021-06-15 17:49:10 +0100 | [diff] [blame] | 202 | // Include all non APEX jars |
Jiakai Zhang | c08c162 | 2023-05-10 18:38:34 +0100 | [diff] [blame] | 203 | jars := b.platformJars(ctx) |
satayev | b309050 | 2021-06-15 17:49:10 +0100 | [diff] [blame] | 204 | |
| 205 | // Include jars from APEXes that don't populate their classpath proto config. |
satayev | d604b21 | 2021-07-21 14:23:52 +0100 | [diff] [blame] | 206 | remainingJars := dexpreopt.GetGlobalConfig(ctx).ApexBootJars |
satayev | b309050 | 2021-06-15 17:49:10 +0100 | [diff] [blame] | 207 | for _, fragment := range b.fragments { |
| 208 | info := ctx.OtherModuleProvider(fragment, ClasspathFragmentProtoContentInfoProvider).(ClasspathFragmentProtoContentInfo) |
| 209 | if info.ClasspathFragmentProtoGenerated { |
| 210 | remainingJars = remainingJars.RemoveList(info.ClasspathFragmentProtoContents) |
| 211 | } |
| 212 | } |
| 213 | for i := 0; i < remainingJars.Len(); i++ { |
| 214 | jars = jars.Append(remainingJars.Apex(i), remainingJars.Jar(i)) |
| 215 | } |
| 216 | |
| 217 | return jars |
satayev | 013485b | 2021-05-06 23:38:10 +0100 | [diff] [blame] | 218 | } |
| 219 | |
Jiakai Zhang | c08c162 | 2023-05-10 18:38:34 +0100 | [diff] [blame] | 220 | func (b *platformBootclasspathModule) platformJars(ctx android.PathContext) android.ConfiguredJarList { |
| 221 | return defaultBootImageConfig(ctx).modules.RemoveList(artBootImageConfig(ctx).modules) |
| 222 | } |
| 223 | |
satayev | d604b21 | 2021-07-21 14:23:52 +0100 | [diff] [blame] | 224 | // checkPlatformModules ensures that the non-updatable modules supplied are not part of an |
| 225 | // apex module. |
| 226 | func (b *platformBootclasspathModule) checkPlatformModules(ctx android.ModuleContext, modules []android.Module) { |
| 227 | // TODO(satayev): change this check to only allow core-icu4j, all apex jars should not be here. |
Paul Duffin | f23bc47 | 2021-04-27 12:42:20 +0100 | [diff] [blame] | 228 | for _, m := range modules { |
| 229 | apexInfo := ctx.OtherModuleProvider(m, android.ApexInfoProvider).(android.ApexInfo) |
| 230 | fromUpdatableApex := apexInfo.Updatable |
| 231 | if fromUpdatableApex { |
| 232 | // error: this jar is part of an updatable apex |
Jiakai Zhang | c08c162 | 2023-05-10 18:38:34 +0100 | [diff] [blame] | 233 | ctx.ModuleErrorf("module %q from updatable apexes %q is not allowed in the platform bootclasspath", ctx.OtherModuleName(m), apexInfo.InApexVariants) |
Paul Duffin | f23bc47 | 2021-04-27 12:42:20 +0100 | [diff] [blame] | 234 | } else { |
| 235 | // ok: this jar is part of the platform or a non-updatable apex |
| 236 | } |
| 237 | } |
| 238 | } |
| 239 | |
satayev | d604b21 | 2021-07-21 14:23:52 +0100 | [diff] [blame] | 240 | // checkApexModules ensures that the apex modules supplied are not from the platform. |
| 241 | func (b *platformBootclasspathModule) checkApexModules(ctx android.ModuleContext, modules []android.Module) { |
Paul Duffin | f23bc47 | 2021-04-27 12:42:20 +0100 | [diff] [blame] | 242 | for _, m := range modules { |
| 243 | apexInfo := ctx.OtherModuleProvider(m, android.ApexInfoProvider).(android.ApexInfo) |
| 244 | fromUpdatableApex := apexInfo.Updatable |
| 245 | if fromUpdatableApex { |
| 246 | // ok: this jar is part of an updatable apex |
| 247 | } else { |
| 248 | name := ctx.OtherModuleName(m) |
| 249 | if apexInfo.IsForPlatform() { |
Paul Duffin | 7487a7a | 2021-05-19 09:36:09 +0100 | [diff] [blame] | 250 | // If AlwaysUsePrebuiltSdks() returns true then it is possible that the updatable list will |
| 251 | // include platform variants of a prebuilt module due to workarounds elsewhere. In that case |
| 252 | // do not treat this as an error. |
| 253 | // TODO(b/179354495): Always treat this as an error when migration to bootclasspath_fragment |
| 254 | // modules is complete. |
| 255 | if !ctx.Config().AlwaysUsePrebuiltSdks() { |
| 256 | // error: this jar is part of the platform |
satayev | d604b21 | 2021-07-21 14:23:52 +0100 | [diff] [blame] | 257 | ctx.ModuleErrorf("module %q from platform is not allowed in the apex boot jars list", name) |
Paul Duffin | 7487a7a | 2021-05-19 09:36:09 +0100 | [diff] [blame] | 258 | } |
Paul Duffin | f23bc47 | 2021-04-27 12:42:20 +0100 | [diff] [blame] | 259 | } else { |
| 260 | // TODO(b/177892522): Treat this as an error. |
| 261 | // Cannot do that at the moment because framework-wifi and framework-tethering are in the |
satayev | d604b21 | 2021-07-21 14:23:52 +0100 | [diff] [blame] | 262 | // PRODUCT_APEX_BOOT_JARS but not marked as updatable in AOSP. |
Paul Duffin | f23bc47 | 2021-04-27 12:42:20 +0100 | [diff] [blame] | 263 | } |
| 264 | } |
| 265 | } |
| 266 | } |
| 267 | |
Paul Duffin | 702210b | 2021-04-08 20:12:41 +0100 | [diff] [blame] | 268 | // generateHiddenAPIBuildActions generates all the hidden API related build rules. |
Paul Duffin | c8ead41 | 2021-06-07 19:28:15 +0100 | [diff] [blame] | 269 | func (b *platformBootclasspathModule) generateHiddenAPIBuildActions(ctx android.ModuleContext, modules []android.Module, fragments []android.Module) bootDexJarByModule { |
Paul Duffin | 702210b | 2021-04-08 20:12:41 +0100 | [diff] [blame] | 270 | |
Paul Duffin | 90b8ad3 | 2021-04-13 12:25:01 +0100 | [diff] [blame] | 271 | // Save the paths to the monolithic files for retrieval via OutputFiles(). |
| 272 | b.hiddenAPIFlagsCSV = hiddenAPISingletonPaths(ctx).flags |
| 273 | b.hiddenAPIIndexCSV = hiddenAPISingletonPaths(ctx).index |
| 274 | b.hiddenAPIMetadataCSV = hiddenAPISingletonPaths(ctx).metadata |
Paul Duffin | 6a76645 | 2021-04-12 14:15:22 +0100 | [diff] [blame] | 275 | |
Adrian Roos | e95a15e | 2021-06-21 16:03:11 +0200 | [diff] [blame] | 276 | bootDexJarByModule := extractBootDexJarsFromModules(ctx, modules) |
| 277 | |
Paul Duffin | 0b65986 | 2021-04-13 13:02:29 +0100 | [diff] [blame] | 278 | // Don't run any hiddenapi rules if UNSAFE_DISABLE_HIDDENAPI_FLAGS=true. This is a performance |
| 279 | // optimization that can be used to reduce the incremental build time but as its name suggests it |
| 280 | // can be unsafe to use, e.g. when the changes affect anything that goes on the bootclasspath. |
Vishnu Nair | 0dbd02a | 2021-04-30 00:24:07 +0000 | [diff] [blame] | 281 | if ctx.Config().IsEnvTrue("UNSAFE_DISABLE_HIDDENAPI_FLAGS") { |
Paul Duffin | 0b65986 | 2021-04-13 13:02:29 +0100 | [diff] [blame] | 282 | paths := android.OutputPaths{b.hiddenAPIFlagsCSV, b.hiddenAPIIndexCSV, b.hiddenAPIMetadataCSV} |
| 283 | for _, path := range paths { |
| 284 | ctx.Build(pctx, android.BuildParams{ |
| 285 | Rule: android.Touch, |
| 286 | Output: path, |
| 287 | }) |
| 288 | } |
Adrian Roos | e95a15e | 2021-06-21 16:03:11 +0200 | [diff] [blame] | 289 | return bootDexJarByModule |
Paul Duffin | 0b65986 | 2021-04-13 13:02:29 +0100 | [diff] [blame] | 290 | } |
| 291 | |
Paul Duffin | 89f570a | 2021-06-16 01:42:33 +0100 | [diff] [blame] | 292 | // Construct a list of ClasspathElement objects from the modules and fragments. |
| 293 | classpathElements := CreateClasspathElements(ctx, modules, fragments) |
| 294 | |
| 295 | monolithicInfo := b.createAndProvideMonolithicHiddenAPIInfo(ctx, classpathElements) |
| 296 | |
| 297 | // Extract the classes jars only from those libraries that do not have corresponding fragments as |
| 298 | // the fragments will have already provided the flags that are needed. |
| 299 | classesJars := monolithicInfo.ClassesJars |
| 300 | |
Paul Duffin | 4539a37 | 2021-06-23 23:20:43 +0100 | [diff] [blame] | 301 | // Create the input to pass to buildRuleToGenerateHiddenAPIStubFlagsFile |
Paul Duffin | 1352f7c | 2021-05-21 22:18:49 +0100 | [diff] [blame] | 302 | input := newHiddenAPIFlagInput() |
| 303 | |
| 304 | // Gather stub library information from the dependencies on modules provided by |
| 305 | // hiddenAPIComputeMonolithicStubLibModules. |
| 306 | input.gatherStubLibInfo(ctx, nil) |
| 307 | |
| 308 | // Use the flag files from this module and all the fragments. |
| 309 | input.FlagFilesByCategory = monolithicInfo.FlagsFilesByCategory |
Paul Duffin | 74431d5 | 2021-04-21 14:10:42 +0100 | [diff] [blame] | 310 | |
Paul Duffin | 537ea3d | 2021-05-14 10:38:00 +0100 | [diff] [blame] | 311 | // Generate the monolithic stub-flags.csv file. |
Paul Duffin | 537ea3d | 2021-05-14 10:38:00 +0100 | [diff] [blame] | 312 | stubFlags := hiddenAPISingletonPaths(ctx).stubFlags |
Paul Duffin | 67b9d61 | 2021-07-21 17:38:47 +0100 | [diff] [blame] | 313 | buildRuleToGenerateHiddenAPIStubFlagsFile(ctx, "platform-bootclasspath-monolithic-hiddenapi-stub-flags", "monolithic hidden API stub flags", stubFlags, bootDexJarByModule.bootDexJars(), input, monolithicInfo.StubFlagSubsets) |
Paul Duffin | 537ea3d | 2021-05-14 10:38:00 +0100 | [diff] [blame] | 314 | |
Paul Duffin | 537ea3d | 2021-05-14 10:38:00 +0100 | [diff] [blame] | 315 | // Generate the annotation-flags.csv file from all the module annotations. |
Paul Duffin | d061d40 | 2021-06-07 21:36:01 +0100 | [diff] [blame] | 316 | annotationFlags := android.PathForModuleOut(ctx, "hiddenapi-monolithic", "annotation-flags-from-classes.csv") |
| 317 | buildRuleToGenerateAnnotationFlags(ctx, "intermediate hidden API flags", classesJars, stubFlags, annotationFlags) |
Paul Duffin | 537ea3d | 2021-05-14 10:38:00 +0100 | [diff] [blame] | 318 | |
Paul Duffin | d061d40 | 2021-06-07 21:36:01 +0100 | [diff] [blame] | 319 | // Generate the monolithic hiddenapi-flags.csv file. |
| 320 | // |
| 321 | // Use annotation flags generated directly from the classes jars as well as annotation flag files |
| 322 | // provided by prebuilts. |
| 323 | allAnnotationFlagFiles := android.Paths{annotationFlags} |
| 324 | allAnnotationFlagFiles = append(allAnnotationFlagFiles, monolithicInfo.AnnotationFlagsPaths...) |
Paul Duffin | 537ea3d | 2021-05-14 10:38:00 +0100 | [diff] [blame] | 325 | allFlags := hiddenAPISingletonPaths(ctx).flags |
Paul Duffin | 67b9d61 | 2021-07-21 17:38:47 +0100 | [diff] [blame] | 326 | buildRuleToGenerateHiddenApiFlags(ctx, "hiddenAPIFlagsFile", "monolithic hidden API flags", allFlags, stubFlags, allAnnotationFlagFiles, monolithicInfo.FlagsFilesByCategory, monolithicInfo.FlagSubsets, android.OptionalPath{}) |
Paul Duffin | 537ea3d | 2021-05-14 10:38:00 +0100 | [diff] [blame] | 327 | |
| 328 | // Generate an intermediate monolithic hiddenapi-metadata.csv file directly from the annotations |
| 329 | // in the source code. |
Paul Duffin | d061d40 | 2021-06-07 21:36:01 +0100 | [diff] [blame] | 330 | intermediateMetadataCSV := android.PathForModuleOut(ctx, "hiddenapi-monolithic", "metadata-from-classes.csv") |
| 331 | buildRuleToGenerateMetadata(ctx, "intermediate hidden API metadata", classesJars, stubFlags, intermediateMetadataCSV) |
Paul Duffin | 537ea3d | 2021-05-14 10:38:00 +0100 | [diff] [blame] | 332 | |
Paul Duffin | d061d40 | 2021-06-07 21:36:01 +0100 | [diff] [blame] | 333 | // Generate the monolithic hiddenapi-metadata.csv file. |
| 334 | // |
| 335 | // Use metadata files generated directly from the classes jars as well as metadata files provided |
| 336 | // by prebuilts. |
| 337 | // |
| 338 | // This has the side effect of ensuring that the output file uses | quotes just in case that is |
| 339 | // important for the tools that consume the metadata file. |
| 340 | allMetadataFlagFiles := android.Paths{intermediateMetadataCSV} |
| 341 | allMetadataFlagFiles = append(allMetadataFlagFiles, monolithicInfo.MetadataPaths...) |
Paul Duffin | 537ea3d | 2021-05-14 10:38:00 +0100 | [diff] [blame] | 342 | metadataCSV := hiddenAPISingletonPaths(ctx).metadata |
Paul Duffin | d061d40 | 2021-06-07 21:36:01 +0100 | [diff] [blame] | 343 | b.buildRuleMergeCSV(ctx, "monolithic hidden API metadata", allMetadataFlagFiles, metadataCSV) |
Paul Duffin | 537ea3d | 2021-05-14 10:38:00 +0100 | [diff] [blame] | 344 | |
Paul Duffin | d061d40 | 2021-06-07 21:36:01 +0100 | [diff] [blame] | 345 | // Generate an intermediate monolithic hiddenapi-index.csv file directly from the CSV files in the |
| 346 | // classes jars. |
| 347 | intermediateIndexCSV := android.PathForModuleOut(ctx, "hiddenapi-monolithic", "index-from-classes.csv") |
| 348 | buildRuleToGenerateIndex(ctx, "intermediate hidden API index", classesJars, intermediateIndexCSV) |
| 349 | |
| 350 | // Generate the monolithic hiddenapi-index.csv file. |
| 351 | // |
| 352 | // Use index files generated directly from the classes jars as well as index files provided |
| 353 | // by prebuilts. |
| 354 | allIndexFlagFiles := android.Paths{intermediateIndexCSV} |
| 355 | allIndexFlagFiles = append(allIndexFlagFiles, monolithicInfo.IndexPaths...) |
Paul Duffin | 537ea3d | 2021-05-14 10:38:00 +0100 | [diff] [blame] | 356 | indexCSV := hiddenAPISingletonPaths(ctx).index |
Paul Duffin | d061d40 | 2021-06-07 21:36:01 +0100 | [diff] [blame] | 357 | b.buildRuleMergeCSV(ctx, "monolithic hidden API index", allIndexFlagFiles, indexCSV) |
Paul Duffin | c8ead41 | 2021-06-07 19:28:15 +0100 | [diff] [blame] | 358 | |
| 359 | return bootDexJarByModule |
Paul Duffin | 74431d5 | 2021-04-21 14:10:42 +0100 | [diff] [blame] | 360 | } |
| 361 | |
Paul Duffin | 438eb57 | 2021-05-21 16:58:23 +0100 | [diff] [blame] | 362 | // createAndProvideMonolithicHiddenAPIInfo creates a MonolithicHiddenAPIInfo and provides it for |
| 363 | // testing. |
Paul Duffin | 89f570a | 2021-06-16 01:42:33 +0100 | [diff] [blame] | 364 | func (b *platformBootclasspathModule) createAndProvideMonolithicHiddenAPIInfo(ctx android.ModuleContext, classpathElements ClasspathElements) MonolithicHiddenAPIInfo { |
Paul Duffin | 1352f7c | 2021-05-21 22:18:49 +0100 | [diff] [blame] | 365 | // Create a temporary input structure in which to collate information provided directly by this |
| 366 | // module, either through properties or direct dependencies. |
| 367 | temporaryInput := newHiddenAPIFlagInput() |
| 368 | |
| 369 | // Create paths to the flag files specified in the properties. |
Paul Duffin | 9b61abb | 2022-07-27 16:16:54 +0000 | [diff] [blame] | 370 | temporaryInput.extractFlagFilesFromProperties(ctx, &b.properties.HiddenAPIFlagFileProperties) |
Paul Duffin | 1352f7c | 2021-05-21 22:18:49 +0100 | [diff] [blame] | 371 | |
| 372 | // Create the monolithic info, by starting with the flag files specified on this and then merging |
| 373 | // in information from all the fragment dependencies of this. |
Paul Duffin | 89f570a | 2021-06-16 01:42:33 +0100 | [diff] [blame] | 374 | monolithicInfo := newMonolithicHiddenAPIInfo(ctx, temporaryInput.FlagFilesByCategory, classpathElements) |
Paul Duffin | 438eb57 | 2021-05-21 16:58:23 +0100 | [diff] [blame] | 375 | |
| 376 | // Store the information for testing. |
Paul Duffin | 524c82c | 2021-06-09 14:39:28 +0100 | [diff] [blame] | 377 | ctx.SetProvider(MonolithicHiddenAPIInfoProvider, monolithicInfo) |
Paul Duffin | 438eb57 | 2021-05-21 16:58:23 +0100 | [diff] [blame] | 378 | return monolithicInfo |
| 379 | } |
| 380 | |
Paul Duffin | 537ea3d | 2021-05-14 10:38:00 +0100 | [diff] [blame] | 381 | func (b *platformBootclasspathModule) buildRuleMergeCSV(ctx android.ModuleContext, desc string, inputPaths android.Paths, outputPath android.WritablePath) { |
Paul Duffin | 00b2bfd | 2021-04-12 17:24:36 +0100 | [diff] [blame] | 382 | rule := android.NewRuleBuilder(pctx, ctx) |
| 383 | rule.Command(). |
| 384 | BuiltTool("merge_csv"). |
| 385 | Flag("--key_field signature"). |
Paul Duffin | 85dee5d | 2021-04-13 00:14:38 +0100 | [diff] [blame] | 386 | FlagWithOutput("--output=", outputPath). |
Paul Duffin | 537ea3d | 2021-05-14 10:38:00 +0100 | [diff] [blame] | 387 | Inputs(inputPaths) |
Paul Duffin | 85dee5d | 2021-04-13 00:14:38 +0100 | [diff] [blame] | 388 | |
Paul Duffin | 537ea3d | 2021-05-14 10:38:00 +0100 | [diff] [blame] | 389 | rule.Build(desc, desc) |
Paul Duffin | 85dee5d | 2021-04-13 00:14:38 +0100 | [diff] [blame] | 390 | } |
Paul Duffin | ad19d38 | 2021-04-26 16:44:00 +0100 | [diff] [blame] | 391 | |
Paul Duffin | 12d29b7 | 2021-04-29 13:50:01 +0100 | [diff] [blame] | 392 | // generateHiddenApiMakeVars generates make variables needed by hidden API related make rules, e.g. |
| 393 | // veridex and run-appcompat. |
| 394 | func (b *platformBootclasspathModule) generateHiddenApiMakeVars(ctx android.MakeVarsContext) { |
| 395 | if ctx.Config().IsEnvTrue("UNSAFE_DISABLE_HIDDENAPI_FLAGS") { |
| 396 | return |
| 397 | } |
| 398 | // INTERNAL_PLATFORM_HIDDENAPI_FLAGS is used by Make rules in art/ and cts/. |
| 399 | ctx.Strict("INTERNAL_PLATFORM_HIDDENAPI_FLAGS", b.hiddenAPIFlagsCSV.String()) |
| 400 | } |
| 401 | |
Paul Duffin | ad19d38 | 2021-04-26 16:44:00 +0100 | [diff] [blame] | 402 | // generateBootImageBuildActions generates ninja rules related to the boot image creation. |
Jiakai Zhang | c08c162 | 2023-05-10 18:38:34 +0100 | [diff] [blame] | 403 | func (b *platformBootclasspathModule) generateBootImageBuildActions(ctx android.ModuleContext) { |
Paul Duffin | ad19d38 | 2021-04-26 16:44:00 +0100 | [diff] [blame] | 404 | // Force the GlobalSoongConfig to be created and cached for use by the dex_bootjars |
| 405 | // GenerateSingletonBuildActions method as it cannot create it for itself. |
| 406 | dexpreopt.GetGlobalSoongConfig(ctx) |
| 407 | |
Paul Duffin | ad19d38 | 2021-04-26 16:44:00 +0100 | [diff] [blame] | 408 | global := dexpreopt.GetGlobalConfig(ctx) |
| 409 | if !shouldBuildBootImages(ctx.Config(), global) { |
| 410 | return |
| 411 | } |
| 412 | |
Jiakai Zhang | d49324d | 2023-02-24 17:05:02 +0000 | [diff] [blame] | 413 | frameworkBootImageConfig := defaultBootImageConfig(ctx) |
| 414 | bootFrameworkProfileRule(ctx, frameworkBootImageConfig) |
Jiakai Zhang | c08c162 | 2023-05-10 18:38:34 +0100 | [diff] [blame] | 415 | b.generateBootImage(ctx, frameworkBootImageName) |
| 416 | b.generateBootImage(ctx, mainlineBootImageName) |
Jiakai Zhang | d49324d | 2023-02-24 17:05:02 +0000 | [diff] [blame] | 417 | dumpOatRules(ctx, frameworkBootImageConfig) |
| 418 | } |
Paul Duffin | 4c09442 | 2021-04-26 20:10:48 +0100 | [diff] [blame] | 419 | |
Jiakai Zhang | c08c162 | 2023-05-10 18:38:34 +0100 | [diff] [blame] | 420 | func (b *platformBootclasspathModule) generateBootImage(ctx android.ModuleContext, imageName string) { |
Jiakai Zhang | d49324d | 2023-02-24 17:05:02 +0000 | [diff] [blame] | 421 | imageConfig := genBootImageConfigs(ctx)[imageName] |
Paul Duffin | 7ebebfd | 2021-04-27 19:36:57 +0100 | [diff] [blame] | 422 | |
Jiakai Zhang | c08c162 | 2023-05-10 18:38:34 +0100 | [diff] [blame] | 423 | modules := b.getModulesForImage(ctx, imageConfig) |
| 424 | |
Jiakai Zhang | d49324d | 2023-02-24 17:05:02 +0000 | [diff] [blame] | 425 | // Copy module dex jars to their predefined locations. |
| 426 | bootDexJarsByModule := extractEncodedDexJarsFromModules(ctx, modules) |
| 427 | copyBootJarsToPredefinedLocations(ctx, bootDexJarsByModule, imageConfig.dexPathsByModule) |
Paul Duffin | 7ebebfd | 2021-04-27 19:36:57 +0100 | [diff] [blame] | 428 | |
Paul Duffin | 2fc82ad | 2021-04-29 23:36:12 +0100 | [diff] [blame] | 429 | // Build a profile for the image config and then use that to build the boot image. |
| 430 | profile := bootImageProfileRule(ctx, imageConfig) |
Paul Duffin | 56afb27 | 2021-07-01 22:04:22 +0100 | [diff] [blame] | 431 | |
Jiakai Zhang | bc698cd | 2023-05-08 16:28:38 +0000 | [diff] [blame] | 432 | // If dexpreopt of boot image jars should be skipped, generate only a profile. |
| 433 | global := dexpreopt.GetGlobalConfig(ctx) |
| 434 | if global.DisablePreoptBootImages { |
| 435 | return |
| 436 | } |
| 437 | |
Paul Duffin | a56be7d | 2021-07-02 13:00:43 +0100 | [diff] [blame] | 438 | // Build boot image files for the android variants. |
Paul Duffin | 9f6ac0b | 2022-10-04 15:36:44 +0100 | [diff] [blame] | 439 | androidBootImageFiles := buildBootImageVariantsForAndroidOs(ctx, imageConfig, profile) |
Paul Duffin | a56be7d | 2021-07-02 13:00:43 +0100 | [diff] [blame] | 440 | |
| 441 | // Zip the android variant boot image files up. |
Paul Duffin | 9f6ac0b | 2022-10-04 15:36:44 +0100 | [diff] [blame] | 442 | buildBootImageZipInPredefinedLocation(ctx, imageConfig, androidBootImageFiles.byArch) |
Paul Duffin | a56be7d | 2021-07-02 13:00:43 +0100 | [diff] [blame] | 443 | |
| 444 | // Build boot image files for the host variants. There are use directly by ART host side tests. |
| 445 | buildBootImageVariantsForBuildOs(ctx, imageConfig, profile) |
Jiakai Zhang | d49324d | 2023-02-24 17:05:02 +0000 | [diff] [blame] | 446 | } |
Paul Duffin | 2fc82ad | 2021-04-29 23:36:12 +0100 | [diff] [blame] | 447 | |
Jiakai Zhang | d49324d | 2023-02-24 17:05:02 +0000 | [diff] [blame] | 448 | // Copy apex module dex jars to their predefined locations. They will be used for dexpreopt for apps. |
| 449 | func (b *platformBootclasspathModule) copyApexBootJarsForAppsDexpreopt(ctx android.ModuleContext, apexModules []android.Module) { |
| 450 | config := GetApexBootConfig(ctx) |
| 451 | apexBootDexJarsByModule := extractEncodedDexJarsFromModules(ctx, apexModules) |
| 452 | copyBootJarsToPredefinedLocations(ctx, apexBootDexJarsByModule, config.dexPathsByModule) |
Paul Duffin | ad19d38 | 2021-04-26 16:44:00 +0100 | [diff] [blame] | 453 | } |
Jiakai Zhang | c08c162 | 2023-05-10 18:38:34 +0100 | [diff] [blame] | 454 | |
| 455 | func (b *platformBootclasspathModule) getModulesForImage(ctx android.ModuleContext, imageConfig *bootImageConfig) []android.Module { |
| 456 | modules := make([]android.Module, 0, imageConfig.modules.Len()) |
| 457 | for i := 0; i < imageConfig.modules.Len(); i++ { |
| 458 | found := false |
| 459 | for _, module := range b.configuredModules { |
| 460 | name := android.RemoveOptionalPrebuiltPrefix(module.Name()) |
| 461 | if name == imageConfig.modules.Jar(i) { |
| 462 | modules = append(modules, module) |
| 463 | found = true |
| 464 | break |
| 465 | } |
| 466 | } |
| 467 | if !found && !ctx.Config().AllowMissingDependencies() { |
| 468 | ctx.ModuleErrorf( |
| 469 | "Boot image '%s' module '%s' not added as a dependency of platform_bootclasspath", |
| 470 | imageConfig.name, |
| 471 | imageConfig.modules.Jar(i)) |
| 472 | return []android.Module{} |
| 473 | } |
| 474 | } |
| 475 | return modules |
| 476 | } |