blob: b6af3bf284ea786a271a7cbc55940e92f943654b [file] [log] [blame]
Colin Crossf24a22a2019-01-31 14:12:44 -08001// Copyright 2019 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
15package java
16
17import (
Paul Duffin1b033f52019-06-10 14:15:04 +010018 "fmt"
19
Colin Crossf24a22a2019-01-31 14:12:44 -080020 "android/soong/android"
21)
22
23func init() {
24 android.RegisterSingletonType("hiddenapi", hiddenAPISingletonFactory)
Artur Satayevb5df8a02020-02-19 16:39:59 +000025 android.RegisterSingletonType("hiddenapi_index", hiddenAPIIndexSingletonFactory)
Paul Duffin1b033f52019-06-10 14:15:04 +010026 android.RegisterModuleType("hiddenapi_flags", hiddenAPIFlagsFactory)
Colin Crossf24a22a2019-01-31 14:12:44 -080027}
28
29type hiddenAPISingletonPathsStruct struct {
Colin Crossf24a22a2019-01-31 14:12:44 -080030 flags android.OutputPath
Artur Satayevb5df8a02020-02-19 16:39:59 +000031 index android.OutputPath
Colin Crossf24a22a2019-01-31 14:12:44 -080032 metadata android.OutputPath
Artur Satayevb5df8a02020-02-19 16:39:59 +000033 stubFlags android.OutputPath
Colin Crossf24a22a2019-01-31 14:12:44 -080034}
35
36var hiddenAPISingletonPathsKey = android.NewOnceKey("hiddenAPISingletonPathsKey")
37
38// hiddenAPISingletonPaths creates all the paths for singleton files the first time it is called, which may be
39// from a ModuleContext that needs to reference a file that will be created by a singleton rule that hasn't
40// yet been created.
41func hiddenAPISingletonPaths(ctx android.PathContext) hiddenAPISingletonPathsStruct {
42 return ctx.Config().Once(hiddenAPISingletonPathsKey, func() interface{} {
43 return hiddenAPISingletonPathsStruct{
Colin Crossf24a22a2019-01-31 14:12:44 -080044 flags: android.PathForOutput(ctx, "hiddenapi", "hiddenapi-flags.csv"),
Artur Satayevb5df8a02020-02-19 16:39:59 +000045 index: android.PathForOutput(ctx, "hiddenapi", "hiddenapi-index.csv"),
Andrei Onea47841972020-08-10 17:23:52 +010046 metadata: android.PathForOutput(ctx, "hiddenapi", "hiddenapi-unsupported.csv"),
Artur Satayevb5df8a02020-02-19 16:39:59 +000047 stubFlags: android.PathForOutput(ctx, "hiddenapi", "hiddenapi-stub-flags.txt"),
Colin Crossf24a22a2019-01-31 14:12:44 -080048 }
49 }).(hiddenAPISingletonPathsStruct)
50}
51
Colin Crossf24a22a2019-01-31 14:12:44 -080052func hiddenAPISingletonFactory() android.Singleton {
Colin Crossed023ec2019-02-19 12:38:45 -080053 return &hiddenAPISingleton{}
Colin Crossf24a22a2019-01-31 14:12:44 -080054}
55
Colin Crossed023ec2019-02-19 12:38:45 -080056type hiddenAPISingleton struct {
57 flags, metadata android.Path
58}
Colin Crossf24a22a2019-01-31 14:12:44 -080059
60// hiddenAPI singleton rules
Colin Crossed023ec2019-02-19 12:38:45 -080061func (h *hiddenAPISingleton) GenerateBuildActions(ctx android.SingletonContext) {
Colin Crossf24a22a2019-01-31 14:12:44 -080062 // Don't run any hiddenapi rules if UNSAFE_DISABLE_HIDDENAPI_FLAGS=true
63 if ctx.Config().IsEnvTrue("UNSAFE_DISABLE_HIDDENAPI_FLAGS") {
64 return
65 }
66
67 stubFlagsRule(ctx)
68
69 // These rules depend on files located in frameworks/base, skip them if running in a tree that doesn't have them.
Jiyong Park09cb6292019-07-15 15:29:23 +090070 if ctx.Config().FrameworksBaseDirExists(ctx) {
Colin Crossed023ec2019-02-19 12:38:45 -080071 h.flags = flagsRule(ctx)
72 h.metadata = metadataRule(ctx)
Colin Crossf24a22a2019-01-31 14:12:44 -080073 } else {
Colin Crossed023ec2019-02-19 12:38:45 -080074 h.flags = emptyFlagsRule(ctx)
75 }
76}
77
78// Export paths to Make. INTERNAL_PLATFORM_HIDDENAPI_FLAGS is used by Make rules in art/ and cts/.
79// Both paths are used to call dist-for-goals.
80func (h *hiddenAPISingleton) MakeVars(ctx android.MakeVarsContext) {
81 if ctx.Config().IsEnvTrue("UNSAFE_DISABLE_HIDDENAPI_FLAGS") {
82 return
83 }
84
85 ctx.Strict("INTERNAL_PLATFORM_HIDDENAPI_FLAGS", h.flags.String())
86
87 if h.metadata != nil {
88 ctx.Strict("INTERNAL_PLATFORM_HIDDENAPI_GREYLIST_METADATA", h.metadata.String())
Colin Crossf24a22a2019-01-31 14:12:44 -080089 }
90}
91
92// stubFlagsRule creates the rule to build hiddenapi-stub-flags.txt out of dex jars from stub modules and boot image
93// modules.
94func stubFlagsRule(ctx android.SingletonContext) {
95 // Public API stubs
96 publicStubModules := []string{
97 "android_stubs_current",
Paul Duffin719fed42019-02-28 16:15:44 +000098 }
99
100 // Add the android.test.base to the set of stubs only if the android.test.base module is on
101 // the boot jars list as the runtime will only enforce hiddenapi access against modules on
102 // that list.
Jeongik Cha816a23a2020-07-08 01:09:23 +0900103 if inList("android.test.base", ctx.Config().BootJars()) && !ctx.Config().AlwaysUsePrebuiltSdks() {
Paul Duffin719fed42019-02-28 16:15:44 +0000104 publicStubModules = append(publicStubModules, "android.test.base.stubs")
Colin Crossf24a22a2019-01-31 14:12:44 -0800105 }
106
107 // System API stubs
108 systemStubModules := []string{
109 "android_system_stubs_current",
110 }
111
112 // Test API stubs
113 testStubModules := []string{
114 "android_test_stubs_current",
115 }
116
117 // Core Platform API stubs
118 corePlatformStubModules := []string{
Pete Gillin1f41dbf2020-06-02 15:59:45 +0100119 "legacy.core.platform.api.stubs",
Colin Crossf24a22a2019-01-31 14:12:44 -0800120 }
121
122 // Allow products to define their own stubs for custom product jars that apps can use.
123 publicStubModules = append(publicStubModules, ctx.Config().ProductHiddenAPIStubs()...)
124 systemStubModules = append(systemStubModules, ctx.Config().ProductHiddenAPIStubsSystem()...)
125 testStubModules = append(testStubModules, ctx.Config().ProductHiddenAPIStubsTest()...)
Allen Hairde816cf2019-02-25 16:37:42 -0800126 if ctx.Config().IsEnvTrue("EMMA_INSTRUMENT") {
127 publicStubModules = append(publicStubModules, "jacoco-stubs")
128 }
Colin Crossf24a22a2019-01-31 14:12:44 -0800129
130 publicStubPaths := make(android.Paths, len(publicStubModules))
131 systemStubPaths := make(android.Paths, len(systemStubModules))
132 testStubPaths := make(android.Paths, len(testStubModules))
133 corePlatformStubPaths := make(android.Paths, len(corePlatformStubModules))
134
135 moduleListToPathList := map[*[]string]android.Paths{
136 &publicStubModules: publicStubPaths,
137 &systemStubModules: systemStubPaths,
138 &testStubModules: testStubPaths,
139 &corePlatformStubModules: corePlatformStubPaths,
140 }
141
142 var bootDexJars android.Paths
143
144 ctx.VisitAllModules(func(module android.Module) {
145 // Collect dex jar paths for the modules listed above.
146 if j, ok := module.(Dependency); ok {
147 name := ctx.ModuleName(module)
148 for moduleList, pathList := range moduleListToPathList {
149 if i := android.IndexList(name, *moduleList); i != -1 {
Ulyana Trafimovich5539e7b2020-06-04 14:08:17 +0000150 pathList[i] = j.DexJarBuildPath()
Colin Crossf24a22a2019-01-31 14:12:44 -0800151 }
152 }
153 }
154
155 // Collect dex jar paths for modules that had hiddenapi encode called on them.
156 if h, ok := module.(hiddenAPIIntf); ok {
157 if jar := h.bootDexJar(); jar != nil {
Jiyong Park4ed468c2019-12-19 02:11:10 +0000158 // For a java lib included in an APEX, only take the one built for
159 // the platform variant, and skip the variants for APEXes.
160 // Otherwise, the hiddenapi tool will complain about duplicated classes
161 if a, ok := module.(android.ApexModule); ok {
162 if android.InAnyApex(module.Name()) && !a.IsForPlatform() {
Jiyong Park7f7766d2019-07-25 22:02:35 +0900163 return
164 }
165 }
Liz Kammer5ca3a622020-08-05 15:40:41 -0700166
Colin Crossf24a22a2019-01-31 14:12:44 -0800167 bootDexJars = append(bootDexJars, jar)
168 }
169 }
170 })
171
172 var missingDeps []string
173 // Ensure all modules were converted to paths
174 for moduleList, pathList := range moduleListToPathList {
175 for i := range pathList {
176 if pathList[i] == nil {
Colin Crosscaa0e1e2019-04-02 13:03:46 -0700177 pathList[i] = android.PathForOutput(ctx, "missing")
Colin Crossf24a22a2019-01-31 14:12:44 -0800178 if ctx.Config().AllowMissingDependencies() {
179 missingDeps = append(missingDeps, (*moduleList)[i])
Colin Crossf24a22a2019-01-31 14:12:44 -0800180 } else {
181 ctx.Errorf("failed to find dex jar path for module %q",
182 (*moduleList)[i])
183 }
184 }
185 }
186 }
187
188 // Singleton rule which applies hiddenapi on all boot class path dex files.
189 rule := android.NewRuleBuilder()
190
191 outputPath := hiddenAPISingletonPaths(ctx).stubFlags
192 tempPath := android.PathForOutput(ctx, outputPath.Rel()+".tmp")
193
194 rule.MissingDeps(missingDeps)
195
196 rule.Command().
Martin Stjernholm7260d062019-12-09 21:47:14 +0000197 Tool(ctx.Config().HostToolPath(ctx, "hiddenapi")).
Colin Crossf24a22a2019-01-31 14:12:44 -0800198 Text("list").
Colin Cross69f59a32019-02-15 10:39:37 -0800199 FlagForEachInput("--boot-dex=", bootDexJars).
200 FlagWithInputList("--public-stub-classpath=", publicStubPaths, ":").
Andrei Oneae04da072019-03-01 17:44:13 +0000201 FlagWithInputList("--system-stub-classpath=", systemStubPaths, ":").
202 FlagWithInputList("--test-stub-classpath=", testStubPaths, ":").
Colin Cross69f59a32019-02-15 10:39:37 -0800203 FlagWithInputList("--core-platform-stub-classpath=", corePlatformStubPaths, ":").
204 FlagWithOutput("--out-api-flags=", tempPath)
Colin Crossf24a22a2019-01-31 14:12:44 -0800205
206 commitChangeForRestat(rule, tempPath, outputPath)
207
208 rule.Build(pctx, ctx, "hiddenAPIStubFlagsFile", "hiddenapi stub flags")
209}
210
Eric Jeong9791efa2020-06-04 17:56:18 -0700211func moduleForGreyListRemovedApis(ctx android.SingletonContext, module android.Module) bool {
212 switch ctx.ModuleName(module) {
213 case "api-stubs-docs", "system-api-stubs-docs", "android.car-stubs-docs", "android.car-system-stubs-docs":
214 return true
215 default:
216 return false
217 }
218}
219
Colin Crossf24a22a2019-01-31 14:12:44 -0800220// flagsRule creates a rule to build hiddenapi-flags.csv out of flags.csv files generated for boot image modules and
Aleksei Kalinovf0f5cdc2020-07-28 13:44:24 +0000221// the unsupported API.
Colin Crossed023ec2019-02-19 12:38:45 -0800222func flagsRule(ctx android.SingletonContext) android.Path {
Colin Crossf24a22a2019-01-31 14:12:44 -0800223 var flagsCSV android.Paths
Artur Satayevc7fb5c92020-03-25 16:48:49 +0000224 var greylistRemovedApis android.Paths
Colin Crossf24a22a2019-01-31 14:12:44 -0800225
226 ctx.VisitAllModules(func(module android.Module) {
227 if h, ok := module.(hiddenAPIIntf); ok {
228 if csv := h.flagsCSV(); csv != nil {
229 flagsCSV = append(flagsCSV, csv)
230 }
Artur Satayevc7fb5c92020-03-25 16:48:49 +0000231 } else if ds, ok := module.(*Droidstubs); ok {
232 // Track @removed public and system APIs via corresponding droidstubs targets.
233 // These APIs are not present in the stubs, however, we have to keep allowing access
234 // to them at runtime.
Eric Jeong9791efa2020-06-04 17:56:18 -0700235 if moduleForGreyListRemovedApis(ctx, module) {
Artur Satayevc7fb5c92020-03-25 16:48:49 +0000236 greylistRemovedApis = append(greylistRemovedApis, ds.removedDexApiFile)
237 }
Colin Crossf24a22a2019-01-31 14:12:44 -0800238 }
239 })
240
Artur Satayevc7fb5c92020-03-25 16:48:49 +0000241 combinedRemovedApis := android.PathForOutput(ctx, "hiddenapi", "combined-removed-dex.txt")
242 ctx.Build(pctx, android.BuildParams{
243 Rule: android.Cat,
244 Inputs: greylistRemovedApis,
245 Output: combinedRemovedApis,
246 Description: "Combine removed apis for " + combinedRemovedApis.String(),
247 })
Colin Crossf24a22a2019-01-31 14:12:44 -0800248
249 rule := android.NewRuleBuilder()
250
251 outputPath := hiddenAPISingletonPaths(ctx).flags
252 tempPath := android.PathForOutput(ctx, outputPath.Rel()+".tmp")
253
254 stubFlags := hiddenAPISingletonPaths(ctx).stubFlags
255
256 rule.Command().
Colin Cross69f59a32019-02-15 10:39:37 -0800257 Tool(android.PathForSource(ctx, "frameworks/base/tools/hiddenapi/generate_hiddenapi_lists.py")).
258 FlagWithInput("--csv ", stubFlags).
259 Inputs(flagsCSV).
Aleksei Kalinovf0f5cdc2020-07-28 13:44:24 +0000260 FlagWithInput("--unsupported ",
Andrei Oneaca790812020-08-04 15:34:35 +0100261 android.PathForSource(ctx, "frameworks/base/config/hiddenapi-unsupported.txt")).
Aleksei Kalinovf0f5cdc2020-07-28 13:44:24 +0000262 FlagWithInput("--unsupported-ignore-conflicts ", combinedRemovedApis).
263 FlagWithInput("--max-target-q ",
Andrei Oneaca790812020-08-04 15:34:35 +0100264 android.PathForSource(ctx, "frameworks/base/config/hiddenapi-max-target-q.txt")).
Aleksei Kalinovf0f5cdc2020-07-28 13:44:24 +0000265 FlagWithInput("--max-target-p ",
Andrei Oneaca790812020-08-04 15:34:35 +0100266 android.PathForSource(ctx, "frameworks/base/config/hiddenapi-max-target-p.txt")).
Aleksei Kalinovf0f5cdc2020-07-28 13:44:24 +0000267 FlagWithInput("--max-target-o-ignore-conflicts ",
Andrei Oneaca790812020-08-04 15:34:35 +0100268 android.PathForSource(ctx, "frameworks/base/config/hiddenapi-max-target-o.txt")).
Aleksei Kalinovf0f5cdc2020-07-28 13:44:24 +0000269 FlagWithInput("--blocked ",
Andrei Oneaca790812020-08-04 15:34:35 +0100270 android.PathForSource(ctx, "frameworks/base/config/hiddenapi-force-blocked.txt")).
Aleksei Kalinovf0f5cdc2020-07-28 13:44:24 +0000271 FlagWithInput("--unsupported-packages ",
Andrei Oneaca790812020-08-04 15:34:35 +0100272 android.PathForSource(ctx, "frameworks/base/config/hiddenapi-unsupported-packages.txt")).
Colin Cross69f59a32019-02-15 10:39:37 -0800273 FlagWithOutput("--output ", tempPath)
Colin Crossf24a22a2019-01-31 14:12:44 -0800274
275 commitChangeForRestat(rule, tempPath, outputPath)
276
277 rule.Build(pctx, ctx, "hiddenAPIFlagsFile", "hiddenapi flags")
Colin Crossed023ec2019-02-19 12:38:45 -0800278
279 return outputPath
Colin Crossf24a22a2019-01-31 14:12:44 -0800280}
281
282// emptyFlagsRule creates a rule to build an empty hiddenapi-flags.csv, which is needed by master-art-host builds that
283// have a partial manifest without frameworks/base but still need to build a boot image.
Colin Crossed023ec2019-02-19 12:38:45 -0800284func emptyFlagsRule(ctx android.SingletonContext) android.Path {
Colin Crossf24a22a2019-01-31 14:12:44 -0800285 rule := android.NewRuleBuilder()
286
287 outputPath := hiddenAPISingletonPaths(ctx).flags
288
Colin Cross69f59a32019-02-15 10:39:37 -0800289 rule.Command().Text("rm").Flag("-f").Output(outputPath)
290 rule.Command().Text("touch").Output(outputPath)
Colin Crossf24a22a2019-01-31 14:12:44 -0800291
292 rule.Build(pctx, ctx, "emptyHiddenAPIFlagsFile", "empty hiddenapi flags")
Colin Crossed023ec2019-02-19 12:38:45 -0800293
294 return outputPath
Colin Crossf24a22a2019-01-31 14:12:44 -0800295}
296
Andrei Onea47841972020-08-10 17:23:52 +0100297// metadataRule creates a rule to build hiddenapi-unsupported.csv out of the metadata.csv files generated for boot image
Colin Crossf24a22a2019-01-31 14:12:44 -0800298// modules.
Colin Crossed023ec2019-02-19 12:38:45 -0800299func metadataRule(ctx android.SingletonContext) android.Path {
Colin Crossf24a22a2019-01-31 14:12:44 -0800300 var metadataCSV android.Paths
301
302 ctx.VisitAllModules(func(module android.Module) {
303 if h, ok := module.(hiddenAPIIntf); ok {
304 if csv := h.metadataCSV(); csv != nil {
305 metadataCSV = append(metadataCSV, csv)
306 }
307 }
308 })
309
310 rule := android.NewRuleBuilder()
311
312 outputPath := hiddenAPISingletonPaths(ctx).metadata
313
314 rule.Command().
Artur Satayevb01dd442020-01-20 17:53:31 +0000315 BuiltTool(ctx, "merge_csv").
Artur Satayev79fac052020-01-20 19:11:33 +0000316 FlagWithOutput("--output=", outputPath).
317 Inputs(metadataCSV)
Colin Crossf24a22a2019-01-31 14:12:44 -0800318
319 rule.Build(pctx, ctx, "hiddenAPIGreylistMetadataFile", "hiddenapi greylist metadata")
Colin Crossed023ec2019-02-19 12:38:45 -0800320
321 return outputPath
Colin Crossf24a22a2019-01-31 14:12:44 -0800322}
323
324// commitChangeForRestat adds a command to a rule that updates outputPath from tempPath if they are different. It
325// also marks the rule as restat and marks the tempPath as a temporary file that should not be considered an output of
326// the rule.
327func commitChangeForRestat(rule *android.RuleBuilder, tempPath, outputPath android.WritablePath) {
328 rule.Restat()
Colin Cross69f59a32019-02-15 10:39:37 -0800329 rule.Temporary(tempPath)
Colin Crossf24a22a2019-01-31 14:12:44 -0800330 rule.Command().
331 Text("(").
332 Text("if").
Colin Cross69f59a32019-02-15 10:39:37 -0800333 Text("cmp -s").Input(tempPath).Output(outputPath).Text(";").
Colin Crossf24a22a2019-01-31 14:12:44 -0800334 Text("then").
Colin Cross69f59a32019-02-15 10:39:37 -0800335 Text("rm").Input(tempPath).Text(";").
Colin Crossf24a22a2019-01-31 14:12:44 -0800336 Text("else").
Colin Cross69f59a32019-02-15 10:39:37 -0800337 Text("mv").Input(tempPath).Output(outputPath).Text(";").
Colin Crossf24a22a2019-01-31 14:12:44 -0800338 Text("fi").
339 Text(")")
340}
Paul Duffin1b033f52019-06-10 14:15:04 +0100341
342type hiddenAPIFlagsProperties struct {
343 // name of the file into which the flags will be copied.
344 Filename *string
345}
346
347type hiddenAPIFlags struct {
348 android.ModuleBase
349
350 properties hiddenAPIFlagsProperties
351
352 outputFilePath android.OutputPath
353}
354
355func (h *hiddenAPIFlags) GenerateAndroidBuildActions(ctx android.ModuleContext) {
356 filename := String(h.properties.Filename)
357
358 inputPath := hiddenAPISingletonPaths(ctx).flags
359 h.outputFilePath = android.PathForModuleOut(ctx, filename).OutputPath
360
361 // This ensures that outputFilePath has the correct name for others to
362 // use, as the source file may have a different name.
363 ctx.Build(pctx, android.BuildParams{
364 Rule: android.Cp,
365 Output: h.outputFilePath,
366 Input: inputPath,
367 })
368}
369
370func (h *hiddenAPIFlags) OutputFiles(tag string) (android.Paths, error) {
371 switch tag {
372 case "":
373 return android.Paths{h.outputFilePath}, nil
374 default:
375 return nil, fmt.Errorf("unsupported module reference tag %q", tag)
376 }
377}
378
379// hiddenapi-flags provides access to the hiddenapi-flags.csv file generated during the build.
380func hiddenAPIFlagsFactory() android.Module {
381 module := &hiddenAPIFlags{}
382 module.AddProperties(&module.properties)
383 android.InitAndroidArchModule(module, android.HostAndDeviceSupported, android.MultilibCommon)
384 return module
385}
Artur Satayevb5df8a02020-02-19 16:39:59 +0000386
387func hiddenAPIIndexSingletonFactory() android.Singleton {
388 return &hiddenAPIIndexSingleton{}
389}
390
391type hiddenAPIIndexSingleton struct {
392 index android.Path
393}
394
395func (h *hiddenAPIIndexSingleton) GenerateBuildActions(ctx android.SingletonContext) {
396 // Don't run any hiddenapi rules if UNSAFE_DISABLE_HIDDENAPI_FLAGS=true
397 if ctx.Config().IsEnvTrue("UNSAFE_DISABLE_HIDDENAPI_FLAGS") {
398 return
399 }
400
401 indexes := android.Paths{}
402 ctx.VisitAllModules(func(module android.Module) {
403 if h, ok := module.(hiddenAPIIntf); ok {
404 if h.indexCSV() != nil {
405 indexes = append(indexes, h.indexCSV())
406 }
407 }
408 })
409
410 rule := android.NewRuleBuilder()
411 rule.Command().
412 BuiltTool(ctx, "merge_csv").
413 FlagWithArg("--header=", "signature,file,startline,startcol,endline,endcol,properties").
414 FlagWithOutput("--output=", hiddenAPISingletonPaths(ctx).index).
415 Inputs(indexes)
416 rule.Build(pctx, ctx, "singleton-merged-hiddenapi-index", "Singleton merged Hidden API index")
417
418 h.index = hiddenAPISingletonPaths(ctx).index
419}
420
421func (h *hiddenAPIIndexSingleton) MakeVars(ctx android.MakeVarsContext) {
422 if ctx.Config().IsEnvTrue("UNSAFE_DISABLE_HIDDENAPI_FLAGS") {
423 return
424 }
425
426 ctx.Strict("INTERNAL_PLATFORM_HIDDENAPI_INDEX", h.index.String())
427}