blob: e453e62d6b4df9466c44a209438b123206256010 [file] [log] [blame]
Colin Cross2207f872021-03-24 12:39:08 -07001// 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
15package java
16
17import (
18 "fmt"
19 "strings"
20
21 "github.com/google/blueprint/proptools"
22
23 "android/soong/android"
24 "android/soong/java/config"
25 "android/soong/remoteexec"
26)
27
28func init() {
29 RegisterStubsBuildComponents(android.InitRegistrationContext)
30}
31
32func RegisterStubsBuildComponents(ctx android.RegistrationContext) {
33 ctx.RegisterModuleType("stubs_defaults", StubsDefaultsFactory)
34
35 ctx.RegisterModuleType("droidstubs", DroidstubsFactory)
36 ctx.RegisterModuleType("droidstubs_host", DroidstubsHostFactory)
37
38 ctx.RegisterModuleType("prebuilt_stubs_sources", PrebuiltStubsSourcesFactory)
39}
40
41//
42// Droidstubs
43//
44type Droidstubs struct {
45 Javadoc
46 android.SdkBase
47
48 properties DroidstubsProperties
49 apiFile android.WritablePath
50 apiXmlFile android.WritablePath
51 lastReleasedApiXmlFile android.WritablePath
52 privateApiFile android.WritablePath
53 removedApiFile android.WritablePath
54 removedDexApiFile android.WritablePath
55 nullabilityWarningsFile android.WritablePath
56
57 checkCurrentApiTimestamp android.WritablePath
58 updateCurrentApiTimestamp android.WritablePath
59 checkLastReleasedApiTimestamp android.WritablePath
60 apiLintTimestamp android.WritablePath
61 apiLintReport android.WritablePath
62
63 checkNullabilityWarningsTimestamp android.WritablePath
64
65 annotationsZip android.WritablePath
66 apiVersionsXml android.WritablePath
67
68 apiFilePath android.Path
69 removedApiFilePath android.Path
70
71 metadataZip android.WritablePath
72 metadataDir android.WritablePath
73}
74
75type DroidstubsProperties struct {
76 // The generated public API filename by Metalava, defaults to <module>_api.txt
77 Api_filename *string
78
79 // the generated removed API filename by Metalava, defaults to <module>_removed.txt
80 Removed_api_filename *string
81
82 // the generated removed Dex API filename by Metalava.
83 Removed_dex_api_filename *string
84
85 Check_api struct {
86 Last_released ApiToCheck
87
88 Current ApiToCheck
89
90 Api_lint struct {
91 Enabled *bool
92
93 // If set, performs api_lint on any new APIs not found in the given signature file
94 New_since *string `android:"path"`
95
96 // If not blank, path to the baseline txt file for approved API lint violations.
97 Baseline_file *string `android:"path"`
98 }
99 }
100
101 // user can specify the version of previous released API file in order to do compatibility check.
102 Previous_api *string `android:"path"`
103
104 // is set to true, Metalava will allow framework SDK to contain annotations.
105 Annotations_enabled *bool
106
107 // a list of top-level directories containing files to merge qualifier annotations (i.e. those intended to be included in the stubs written) from.
108 Merge_annotations_dirs []string
109
110 // a list of top-level directories containing Java stub files to merge show/hide annotations from.
111 Merge_inclusion_annotations_dirs []string
112
113 // a file containing a list of classes to do nullability validation for.
114 Validate_nullability_from_list *string
115
116 // a file containing expected warnings produced by validation of nullability annotations.
117 Check_nullability_warnings *string
118
119 // if set to true, allow Metalava to generate doc_stubs source files. Defaults to false.
120 Create_doc_stubs *bool
121
122 // if set to true, cause Metalava to output Javadoc comments in the stubs source files. Defaults to false.
123 // Has no effect if create_doc_stubs: true.
124 Output_javadoc_comments *bool
125
126 // if set to false then do not write out stubs. Defaults to true.
127 //
128 // TODO(b/146727827): Remove capability when we do not need to generate stubs and API separately.
129 Generate_stubs *bool
130
131 // if set to true, provides a hint to the build system that this rule uses a lot of memory,
132 // whicih can be used for scheduling purposes
133 High_mem *bool
134
135 // is set to true, Metalava will allow framework SDK to contain API levels annotations.
136 Api_levels_annotations_enabled *bool
137
138 // the dirs which Metalava extracts API levels annotations from.
139 Api_levels_annotations_dirs []string
140
141 // the filename which Metalava extracts API levels annotations from. Defaults to android.jar.
142 Api_levels_jar_filename *string
143
144 // if set to true, collect the values used by the Dev tools and
145 // write them in files packaged with the SDK. Defaults to false.
146 Write_sdk_values *bool
147}
148
149// droidstubs passes sources files through Metalava to generate stub .java files that only contain the API to be
150// documented, filtering out hidden classes and methods. The resulting .java files are intended to be passed to
151// a droiddoc module to generate documentation.
152func DroidstubsFactory() android.Module {
153 module := &Droidstubs{}
154
155 module.AddProperties(&module.properties,
156 &module.Javadoc.properties)
157
158 InitDroiddocModule(module, android.HostAndDeviceSupported)
159 android.InitSdkAwareModule(module)
160 return module
161}
162
163// droidstubs_host passes sources files through Metalava to generate stub .java files that only contain the API
164// to be documented, filtering out hidden classes and methods. The resulting .java files are intended to be
165// passed to a droiddoc_host module to generate documentation. Use a droidstubs_host instead of a droidstubs
166// module when symbols needed by the source files are provided by java_library_host modules.
167func DroidstubsHostFactory() android.Module {
168 module := &Droidstubs{}
169
170 module.AddProperties(&module.properties,
171 &module.Javadoc.properties)
172
173 InitDroiddocModule(module, android.HostSupported)
174 return module
175}
176
177func (d *Droidstubs) OutputFiles(tag string) (android.Paths, error) {
178 switch tag {
179 case "":
180 return android.Paths{d.stubsSrcJar}, nil
181 case ".docs.zip":
182 return android.Paths{d.docZip}, nil
183 case ".api.txt", android.DefaultDistTag:
184 // This is the default dist path for dist properties that have no tag property.
185 return android.Paths{d.apiFilePath}, nil
186 case ".removed-api.txt":
187 return android.Paths{d.removedApiFilePath}, nil
188 case ".annotations.zip":
189 return android.Paths{d.annotationsZip}, nil
190 case ".api_versions.xml":
191 return android.Paths{d.apiVersionsXml}, nil
192 default:
193 return nil, fmt.Errorf("unsupported module reference tag %q", tag)
194 }
195}
196
197func (d *Droidstubs) ApiFilePath() android.Path {
198 return d.apiFilePath
199}
200
201func (d *Droidstubs) RemovedApiFilePath() android.Path {
202 return d.removedApiFilePath
203}
204
205func (d *Droidstubs) StubsSrcJar() android.Path {
206 return d.stubsSrcJar
207}
208
209var metalavaMergeAnnotationsDirTag = dependencyTag{name: "metalava-merge-annotations-dir"}
210var metalavaMergeInclusionAnnotationsDirTag = dependencyTag{name: "metalava-merge-inclusion-annotations-dir"}
211var metalavaAPILevelsAnnotationsDirTag = dependencyTag{name: "metalava-api-levels-annotations-dir"}
212
213func (d *Droidstubs) DepsMutator(ctx android.BottomUpMutatorContext) {
214 d.Javadoc.addDeps(ctx)
215
216 if len(d.properties.Merge_annotations_dirs) != 0 {
217 for _, mergeAnnotationsDir := range d.properties.Merge_annotations_dirs {
218 ctx.AddDependency(ctx.Module(), metalavaMergeAnnotationsDirTag, mergeAnnotationsDir)
219 }
220 }
221
222 if len(d.properties.Merge_inclusion_annotations_dirs) != 0 {
223 for _, mergeInclusionAnnotationsDir := range d.properties.Merge_inclusion_annotations_dirs {
224 ctx.AddDependency(ctx.Module(), metalavaMergeInclusionAnnotationsDirTag, mergeInclusionAnnotationsDir)
225 }
226 }
227
228 if len(d.properties.Api_levels_annotations_dirs) != 0 {
229 for _, apiLevelsAnnotationsDir := range d.properties.Api_levels_annotations_dirs {
230 ctx.AddDependency(ctx.Module(), metalavaAPILevelsAnnotationsDirTag, apiLevelsAnnotationsDir)
231 }
232 }
233}
234
235func (d *Droidstubs) stubsFlags(ctx android.ModuleContext, cmd *android.RuleBuilderCommand, stubsDir android.OptionalPath) {
236 if apiCheckEnabled(ctx, d.properties.Check_api.Current, "current") ||
237 apiCheckEnabled(ctx, d.properties.Check_api.Last_released, "last_released") ||
238 String(d.properties.Api_filename) != "" {
239 filename := proptools.StringDefault(d.properties.Api_filename, ctx.ModuleName()+"_api.txt")
Colin Crosscb77f752021-03-24 12:04:44 -0700240 d.apiFile = android.PathForModuleOut(ctx, "metalava", filename)
Colin Cross2207f872021-03-24 12:39:08 -0700241 cmd.FlagWithOutput("--api ", d.apiFile)
242 d.apiFilePath = d.apiFile
243 } else if sourceApiFile := proptools.String(d.properties.Check_api.Current.Api_file); sourceApiFile != "" {
244 // If check api is disabled then make the source file available for export.
245 d.apiFilePath = android.PathForModuleSrc(ctx, sourceApiFile)
246 }
247
248 if apiCheckEnabled(ctx, d.properties.Check_api.Current, "current") ||
249 apiCheckEnabled(ctx, d.properties.Check_api.Last_released, "last_released") ||
250 String(d.properties.Removed_api_filename) != "" {
251 filename := proptools.StringDefault(d.properties.Removed_api_filename, ctx.ModuleName()+"_removed.txt")
Colin Crosscb77f752021-03-24 12:04:44 -0700252 d.removedApiFile = android.PathForModuleOut(ctx, "metalava", filename)
Colin Cross2207f872021-03-24 12:39:08 -0700253 cmd.FlagWithOutput("--removed-api ", d.removedApiFile)
254 d.removedApiFilePath = d.removedApiFile
255 } else if sourceRemovedApiFile := proptools.String(d.properties.Check_api.Current.Removed_api_file); sourceRemovedApiFile != "" {
256 // If check api is disabled then make the source removed api file available for export.
257 d.removedApiFilePath = android.PathForModuleSrc(ctx, sourceRemovedApiFile)
258 }
259
260 if String(d.properties.Removed_dex_api_filename) != "" {
Colin Crosscb77f752021-03-24 12:04:44 -0700261 d.removedDexApiFile = android.PathForModuleOut(ctx, "metalava", String(d.properties.Removed_dex_api_filename))
Colin Cross2207f872021-03-24 12:39:08 -0700262 cmd.FlagWithOutput("--removed-dex-api ", d.removedDexApiFile)
263 }
264
265 if Bool(d.properties.Write_sdk_values) {
Colin Crosscb77f752021-03-24 12:04:44 -0700266 d.metadataDir = android.PathForModuleOut(ctx, "metalava", "metadata")
Colin Cross2207f872021-03-24 12:39:08 -0700267 cmd.FlagWithArg("--sdk-values ", d.metadataDir.String())
268 }
269
270 if stubsDir.Valid() {
271 if Bool(d.properties.Create_doc_stubs) {
272 cmd.FlagWithArg("--doc-stubs ", stubsDir.String())
273 } else {
274 cmd.FlagWithArg("--stubs ", stubsDir.String())
275 if !Bool(d.properties.Output_javadoc_comments) {
276 cmd.Flag("--exclude-documentation-from-stubs")
277 }
278 }
279 }
280}
281
282func (d *Droidstubs) annotationsFlags(ctx android.ModuleContext, cmd *android.RuleBuilderCommand) {
283 if Bool(d.properties.Annotations_enabled) {
284 cmd.Flag("--include-annotations")
285
286 validatingNullability :=
287 android.InList("--validate-nullability-from-merged-stubs", d.Javadoc.args) ||
288 String(d.properties.Validate_nullability_from_list) != ""
289
290 migratingNullability := String(d.properties.Previous_api) != ""
291 if migratingNullability {
292 previousApi := android.PathForModuleSrc(ctx, String(d.properties.Previous_api))
293 cmd.FlagWithInput("--migrate-nullness ", previousApi)
294 }
295
296 if s := String(d.properties.Validate_nullability_from_list); s != "" {
297 cmd.FlagWithInput("--validate-nullability-from-list ", android.PathForModuleSrc(ctx, s))
298 }
299
300 if validatingNullability {
Colin Crosscb77f752021-03-24 12:04:44 -0700301 d.nullabilityWarningsFile = android.PathForModuleOut(ctx, "metalava", ctx.ModuleName()+"_nullability_warnings.txt")
Colin Cross2207f872021-03-24 12:39:08 -0700302 cmd.FlagWithOutput("--nullability-warnings-txt ", d.nullabilityWarningsFile)
303 }
304
Colin Crosscb77f752021-03-24 12:04:44 -0700305 d.annotationsZip = android.PathForModuleOut(ctx, "metalava", ctx.ModuleName()+"_annotations.zip")
Colin Cross2207f872021-03-24 12:39:08 -0700306 cmd.FlagWithOutput("--extract-annotations ", d.annotationsZip)
307
308 if len(d.properties.Merge_annotations_dirs) != 0 {
309 d.mergeAnnoDirFlags(ctx, cmd)
310 }
311
312 // TODO(tnorbye): find owners to fix these warnings when annotation was enabled.
313 cmd.FlagWithArg("--hide ", "HiddenTypedefConstant").
314 FlagWithArg("--hide ", "SuperfluousPrefix").
315 FlagWithArg("--hide ", "AnnotationExtraction")
316 }
317}
318
319func (d *Droidstubs) mergeAnnoDirFlags(ctx android.ModuleContext, cmd *android.RuleBuilderCommand) {
320 ctx.VisitDirectDepsWithTag(metalavaMergeAnnotationsDirTag, func(m android.Module) {
321 if t, ok := m.(*ExportedDroiddocDir); ok {
322 cmd.FlagWithArg("--merge-qualifier-annotations ", t.dir.String()).Implicits(t.deps)
323 } else {
324 ctx.PropertyErrorf("merge_annotations_dirs",
325 "module %q is not a metalava merge-annotations dir", ctx.OtherModuleName(m))
326 }
327 })
328}
329
330func (d *Droidstubs) inclusionAnnotationsFlags(ctx android.ModuleContext, cmd *android.RuleBuilderCommand) {
331 ctx.VisitDirectDepsWithTag(metalavaMergeInclusionAnnotationsDirTag, func(m android.Module) {
332 if t, ok := m.(*ExportedDroiddocDir); ok {
333 cmd.FlagWithArg("--merge-inclusion-annotations ", t.dir.String()).Implicits(t.deps)
334 } else {
335 ctx.PropertyErrorf("merge_inclusion_annotations_dirs",
336 "module %q is not a metalava merge-annotations dir", ctx.OtherModuleName(m))
337 }
338 })
339}
340
341func (d *Droidstubs) apiLevelsAnnotationsFlags(ctx android.ModuleContext, cmd *android.RuleBuilderCommand) {
342 if !Bool(d.properties.Api_levels_annotations_enabled) {
343 return
344 }
345
Colin Crosscb77f752021-03-24 12:04:44 -0700346 d.apiVersionsXml = android.PathForModuleOut(ctx, "metalava", "api-versions.xml")
Colin Cross2207f872021-03-24 12:39:08 -0700347
348 if len(d.properties.Api_levels_annotations_dirs) == 0 {
349 ctx.PropertyErrorf("api_levels_annotations_dirs",
350 "has to be non-empty if api levels annotations was enabled!")
351 }
352
353 cmd.FlagWithOutput("--generate-api-levels ", d.apiVersionsXml)
354 cmd.FlagWithInput("--apply-api-levels ", d.apiVersionsXml)
355 cmd.FlagWithArg("--current-version ", ctx.Config().PlatformSdkVersion().String())
356 cmd.FlagWithArg("--current-codename ", ctx.Config().PlatformSdkCodename())
357
358 filename := proptools.StringDefault(d.properties.Api_levels_jar_filename, "android.jar")
359
360 ctx.VisitDirectDepsWithTag(metalavaAPILevelsAnnotationsDirTag, func(m android.Module) {
361 if t, ok := m.(*ExportedDroiddocDir); ok {
362 for _, dep := range t.deps {
363 if strings.HasSuffix(dep.String(), filename) {
364 cmd.Implicit(dep)
365 }
366 }
367 cmd.FlagWithArg("--android-jar-pattern ", t.dir.String()+"/%/public/"+filename)
368 } else {
369 ctx.PropertyErrorf("api_levels_annotations_dirs",
370 "module %q is not a metalava api-levels-annotations dir", ctx.OtherModuleName(m))
371 }
372 })
373}
374
375func metalavaCmd(ctx android.ModuleContext, rule *android.RuleBuilder, javaVersion javaVersion, srcs android.Paths,
376 srcJarList android.Path, bootclasspath, classpath classpath, sourcepaths android.Paths,
377 implicitsRsp, homeDir android.WritablePath, sandbox bool) *android.RuleBuilderCommand {
378 rule.Command().Text("rm -rf").Flag(homeDir.String())
379 rule.Command().Text("mkdir -p").Flag(homeDir.String())
380
381 cmd := rule.Command()
382 cmd.FlagWithArg("ANDROID_PREFS_ROOT=", homeDir.String())
383
384 if ctx.Config().UseRBE() && ctx.Config().IsEnvTrue("RBE_METALAVA") {
385 rule.Remoteable(android.RemoteRuleSupports{RBE: true})
Colin Cross6aa5c402021-03-24 12:28:50 -0700386 if sandbox {
387 execStrategy := ctx.Config().GetenvWithDefault("RBE_METALAVA_EXEC_STRATEGY", remoteexec.LocalExecStrategy)
388 labels := map[string]string{"type": "tool", "name": "metalava"}
389 // TODO: metalava pool rejects these jobs
390 pool := ctx.Config().GetenvWithDefault("RBE_METALAVA_POOL", "java16")
391 rule.Rewrapper(&remoteexec.REParams{
392 Labels: labels,
393 ExecStrategy: execStrategy,
394 ToolchainInputs: []string{config.JavaCmd(ctx).String()},
395 Platform: map[string]string{remoteexec.PoolKey: pool},
396 })
397 } else {
398 execStrategy := remoteexec.LocalExecStrategy
399 labels := map[string]string{"type": "compile", "lang": "java", "compiler": "metalava", "shallow": "true"}
400 pool := ctx.Config().GetenvWithDefault("RBE_METALAVA_POOL", "metalava")
401
402 inputs := []string{
403 ctx.Config().HostJavaToolPath(ctx, "metalava").String(),
404 homeDir.String(),
405 }
406 if v := ctx.Config().Getenv("RBE_METALAVA_INPUTS"); v != "" {
407 inputs = append(inputs, strings.Split(v, ",")...)
408 }
409 cmd.Text((&remoteexec.REParams{
410 Labels: labels,
411 ExecStrategy: execStrategy,
412 Inputs: inputs,
413 RSPFiles: []string{implicitsRsp.String()},
414 ToolchainInputs: []string{config.JavaCmd(ctx).String()},
415 Platform: map[string]string{remoteexec.PoolKey: pool},
416 EnvironmentVariables: []string{"ANDROID_PREFS_ROOT"},
417 }).NoVarTemplate(ctx.Config().RBEWrapper()))
Colin Cross2207f872021-03-24 12:39:08 -0700418 }
Colin Cross2207f872021-03-24 12:39:08 -0700419 }
420
Colin Cross6aa5c402021-03-24 12:28:50 -0700421 cmd.BuiltTool("metalava").ImplicitTool(ctx.Config().HostJavaToolPath(ctx, "metalava.jar")).
Colin Cross2207f872021-03-24 12:39:08 -0700422 Flag(config.JavacVmFlags).
423 Flag("-J--add-opens=java.base/java.util=ALL-UNNAMED").
424 FlagWithArg("-encoding ", "UTF-8").
425 FlagWithArg("-source ", javaVersion.String()).
426 FlagWithRspFileInputList("@", android.PathForModuleOut(ctx, "metalava.rsp"), srcs).
427 FlagWithInput("@", srcJarList)
428
Colin Cross6aa5c402021-03-24 12:28:50 -0700429 if !sandbox {
430 if javaHome := ctx.Config().Getenv("ANDROID_JAVA_HOME"); javaHome != "" {
431 cmd.Implicit(android.PathForSource(ctx, javaHome))
432 }
Colin Cross2207f872021-03-24 12:39:08 -0700433
Colin Cross2207f872021-03-24 12:39:08 -0700434 cmd.FlagWithOutput("--strict-input-files:warn ", android.PathForModuleOut(ctx, ctx.ModuleName()+"-"+"violations.txt"))
Colin Cross2207f872021-03-24 12:39:08 -0700435
Colin Cross6aa5c402021-03-24 12:28:50 -0700436 if implicitsRsp != nil {
437 cmd.FlagWithArg("--strict-input-files-exempt ", "@"+implicitsRsp.String())
438 }
Colin Cross2207f872021-03-24 12:39:08 -0700439 }
440
441 if len(bootclasspath) > 0 {
442 cmd.FlagWithInputList("-bootclasspath ", bootclasspath.Paths(), ":")
443 }
444
445 if len(classpath) > 0 {
446 cmd.FlagWithInputList("-classpath ", classpath.Paths(), ":")
447 }
448
449 if len(sourcepaths) > 0 {
450 cmd.FlagWithList("-sourcepath ", sourcepaths.Strings(), ":")
451 } else {
452 cmd.FlagWithArg("-sourcepath ", `""`)
453 }
454
455 cmd.Flag("--no-banner").
456 Flag("--color").
457 Flag("--quiet").
458 Flag("--format=v2").
459 FlagWithArg("--repeat-errors-max ", "10").
460 FlagWithArg("--hide ", "UnresolvedImport")
461
462 return cmd
463}
464
465func (d *Droidstubs) GenerateAndroidBuildActions(ctx android.ModuleContext) {
466 deps := d.Javadoc.collectDeps(ctx)
467
468 javaVersion := getJavaVersion(ctx, String(d.Javadoc.properties.Java_version), sdkContext(d))
469
470 // Create rule for metalava
471
Colin Crosscb77f752021-03-24 12:04:44 -0700472 srcJarDir := android.PathForModuleOut(ctx, "metalava", "srcjars")
Colin Cross2207f872021-03-24 12:39:08 -0700473
474 rule := android.NewRuleBuilder(pctx, ctx)
475
Colin Cross6aa5c402021-03-24 12:28:50 -0700476 sandbox := proptools.Bool(d.Javadoc.properties.Sandbox)
477 if sandbox {
478 rule.Sbox(android.PathForModuleOut(ctx, "metalava"),
479 android.PathForModuleOut(ctx, "metalava.sbox.textproto")).
480 SandboxInputs()
481 }
482
Colin Cross2207f872021-03-24 12:39:08 -0700483 if BoolDefault(d.properties.High_mem, false) {
484 // This metalava run uses lots of memory, restrict the number of metalava jobs that can run in parallel.
485 rule.HighMem()
486 }
487
488 generateStubs := BoolDefault(d.properties.Generate_stubs, true)
489 var stubsDir android.OptionalPath
490 if generateStubs {
Colin Crosscb77f752021-03-24 12:04:44 -0700491 d.Javadoc.stubsSrcJar = android.PathForModuleOut(ctx, "metalava", ctx.ModuleName()+"-"+"stubs.srcjar")
492 stubsDir = android.OptionalPathForPath(android.PathForModuleOut(ctx, "metalava", "stubsDir"))
Colin Cross2207f872021-03-24 12:39:08 -0700493 rule.Command().Text("rm -rf").Text(stubsDir.String())
494 rule.Command().Text("mkdir -p").Text(stubsDir.String())
495 }
496
497 srcJarList := zipSyncCmd(ctx, rule, srcJarDir, d.Javadoc.srcJars)
498
499 implicitsRsp := android.PathForModuleOut(ctx, ctx.ModuleName()+"-"+"implicits.rsp")
Colin Crosscb77f752021-03-24 12:04:44 -0700500 homeDir := android.PathForModuleOut(ctx, "metalava", "home")
Colin Cross2207f872021-03-24 12:39:08 -0700501 cmd := metalavaCmd(ctx, rule, javaVersion, d.Javadoc.srcFiles, srcJarList,
502 deps.bootClasspath, deps.classpath, d.Javadoc.sourcepaths, implicitsRsp, homeDir,
Colin Cross6aa5c402021-03-24 12:28:50 -0700503 sandbox)
Colin Cross2207f872021-03-24 12:39:08 -0700504 cmd.Implicits(d.Javadoc.implicits)
505
506 d.stubsFlags(ctx, cmd, stubsDir)
507
508 d.annotationsFlags(ctx, cmd)
509 d.inclusionAnnotationsFlags(ctx, cmd)
510 d.apiLevelsAnnotationsFlags(ctx, cmd)
511
512 if android.InList("--generate-documentation", d.Javadoc.args) {
Colin Cross0d532412021-03-25 09:38:45 -0700513 // Currently Metalava have the ability to invoke Javadoc in a separate process.
Colin Cross2207f872021-03-24 12:39:08 -0700514 // Pass "-nodocs" to suppress the Javadoc invocation when Metalava receives
515 // "--generate-documentation" arg. This is not needed when Metalava removes this feature.
516 d.Javadoc.args = append(d.Javadoc.args, "-nodocs")
517 }
518
519 cmd.Flag(strings.Join(d.Javadoc.args, " ")).Implicits(d.Javadoc.argFiles)
520 for _, o := range d.Javadoc.properties.Out {
521 cmd.ImplicitOutput(android.PathForModuleGen(ctx, o))
522 }
523
524 // Add options for the other optional tasks: API-lint and check-released.
525 // We generate separate timestamp files for them.
526
527 doApiLint := false
528 doCheckReleased := false
529
530 // Add API lint options.
531
532 if BoolDefault(d.properties.Check_api.Api_lint.Enabled, false) {
533 doApiLint = true
534
535 newSince := android.OptionalPathForModuleSrc(ctx, d.properties.Check_api.Api_lint.New_since)
536 if newSince.Valid() {
537 cmd.FlagWithInput("--api-lint ", newSince.Path())
538 } else {
539 cmd.Flag("--api-lint")
540 }
Colin Crosscb77f752021-03-24 12:04:44 -0700541 d.apiLintReport = android.PathForModuleOut(ctx, "metalava", "api_lint_report.txt")
Colin Cross2207f872021-03-24 12:39:08 -0700542 cmd.FlagWithOutput("--report-even-if-suppressed ", d.apiLintReport) // TODO: Change to ":api-lint"
543
Colin Cross0d532412021-03-25 09:38:45 -0700544 // TODO(b/154317059): Clean up this allowlist by baselining and/or checking in last-released.
Colin Cross2207f872021-03-24 12:39:08 -0700545 if d.Name() != "android.car-system-stubs-docs" &&
546 d.Name() != "android.car-stubs-docs" {
547 cmd.Flag("--lints-as-errors")
548 cmd.Flag("--warnings-as-errors") // Most lints are actually warnings.
549 }
550
551 baselineFile := android.OptionalPathForModuleSrc(ctx, d.properties.Check_api.Api_lint.Baseline_file)
Colin Crosscb77f752021-03-24 12:04:44 -0700552 updatedBaselineOutput := android.PathForModuleOut(ctx, "metalava", "api_lint_baseline.txt")
553 d.apiLintTimestamp = android.PathForModuleOut(ctx, "metalava", "api_lint.timestamp")
Colin Cross2207f872021-03-24 12:39:08 -0700554
555 // Note this string includes a special shell quote $' ... ', which decodes the "\n"s.
556 // However, because $' ... ' doesn't expand environmental variables, we can't just embed
557 // $PWD, so we have to terminate $'...', use "$PWD", then start $' ... ' again,
558 // which is why we have '"$PWD"$' in it.
559 //
560 // TODO: metalava also has a slightly different message hardcoded. Should we unify this
561 // message and metalava's one?
562 msg := `$'` + // Enclose with $' ... '
563 `************************************************************\n` +
564 `Your API changes are triggering API Lint warnings or errors.\n` +
565 `To make these errors go away, fix the code according to the\n` +
566 `error and/or warning messages above.\n` +
567 `\n` +
568 `If it is not possible to do so, there are workarounds:\n` +
569 `\n` +
570 `1. You can suppress the errors with @SuppressLint("<id>")\n`
571
572 if baselineFile.Valid() {
573 cmd.FlagWithInput("--baseline:api-lint ", baselineFile.Path())
574 cmd.FlagWithOutput("--update-baseline:api-lint ", updatedBaselineOutput)
575
576 msg += fmt.Sprintf(``+
577 `2. You can update the baseline by executing the following\n`+
578 ` command:\n`+
579 ` cp \\\n`+
580 ` "'"$PWD"$'/%s" \\\n`+
581 ` "'"$PWD"$'/%s"\n`+
582 ` To submit the revised baseline.txt to the main Android\n`+
583 ` repository, you will need approval.\n`, updatedBaselineOutput, baselineFile.Path())
584 } else {
585 msg += fmt.Sprintf(``+
586 `2. You can add a baseline file of existing lint failures\n`+
587 ` to the build rule of %s.\n`, d.Name())
588 }
589 // Note the message ends with a ' (single quote), to close the $' ... ' .
590 msg += `************************************************************\n'`
591
592 cmd.FlagWithArg("--error-message:api-lint ", msg)
593 }
594
595 // Add "check released" options. (Detect incompatible API changes from the last public release)
596
597 if apiCheckEnabled(ctx, d.properties.Check_api.Last_released, "last_released") {
598 doCheckReleased = true
599
600 if len(d.Javadoc.properties.Out) > 0 {
601 ctx.PropertyErrorf("out", "out property may not be combined with check_api")
602 }
603
604 apiFile := android.PathForModuleSrc(ctx, String(d.properties.Check_api.Last_released.Api_file))
605 removedApiFile := android.PathForModuleSrc(ctx, String(d.properties.Check_api.Last_released.Removed_api_file))
606 baselineFile := android.OptionalPathForModuleSrc(ctx, d.properties.Check_api.Last_released.Baseline_file)
Colin Crosscb77f752021-03-24 12:04:44 -0700607 updatedBaselineOutput := android.PathForModuleOut(ctx, "metalava", "last_released_baseline.txt")
Colin Cross2207f872021-03-24 12:39:08 -0700608
Colin Crosscb77f752021-03-24 12:04:44 -0700609 d.checkLastReleasedApiTimestamp = android.PathForModuleOut(ctx, "metalava", "check_last_released_api.timestamp")
Colin Cross2207f872021-03-24 12:39:08 -0700610
611 cmd.FlagWithInput("--check-compatibility:api:released ", apiFile)
612 cmd.FlagWithInput("--check-compatibility:removed:released ", removedApiFile)
613
614 if baselineFile.Valid() {
615 cmd.FlagWithInput("--baseline:compatibility:released ", baselineFile.Path())
616 cmd.FlagWithOutput("--update-baseline:compatibility:released ", updatedBaselineOutput)
617 }
618
619 // Note this string includes quote ($' ... '), which decodes the "\n"s.
620 msg := `$'\n******************************\n` +
621 `You have tried to change the API from what has been previously released in\n` +
622 `an SDK. Please fix the errors listed above.\n` +
623 `******************************\n'`
624
625 cmd.FlagWithArg("--error-message:compatibility:released ", msg)
626 }
627
Colin Cross6aa5c402021-03-24 12:28:50 -0700628 if !sandbox {
629 // When sandboxing is enabled RuleBuilder tracks all the inputs needed for remote execution.
630 // Without it we have to do it manually.
631 impRule := android.NewRuleBuilder(pctx, ctx)
632 impCmd := impRule.Command()
633 // An action that copies the ninja generated rsp file to a new location. This allows us to
634 // add a large number of inputs to a file without exceeding bash command length limits (which
635 // would happen if we use the WriteFile rule). The cp is needed because RuleBuilder sets the
636 // rsp file to be ${output}.rsp.
637 impCmd.Text("cp").
638 FlagWithRspFileInputList("", android.PathForModuleOut(ctx, "metalava-implicits.rsp"), cmd.GetImplicits()).
639 Output(implicitsRsp)
640 impRule.Build("implicitsGen", "implicits generation")
641 cmd.Implicit(implicitsRsp)
642 }
Colin Cross2207f872021-03-24 12:39:08 -0700643
644 if generateStubs {
645 rule.Command().
646 BuiltTool("soong_zip").
647 Flag("-write_if_changed").
648 Flag("-jar").
649 FlagWithOutput("-o ", d.Javadoc.stubsSrcJar).
650 FlagWithArg("-C ", stubsDir.String()).
651 FlagWithArg("-D ", stubsDir.String())
652 }
653
654 if Bool(d.properties.Write_sdk_values) {
Colin Crosscb77f752021-03-24 12:04:44 -0700655 d.metadataZip = android.PathForModuleOut(ctx, "metalava", ctx.ModuleName()+"-metadata.zip")
Colin Cross2207f872021-03-24 12:39:08 -0700656 rule.Command().
657 BuiltTool("soong_zip").
658 Flag("-write_if_changed").
659 Flag("-d").
660 FlagWithOutput("-o ", d.metadataZip).
661 FlagWithArg("-C ", d.metadataDir.String()).
662 FlagWithArg("-D ", d.metadataDir.String())
663 }
664
665 // TODO: We don't really need two separate API files, but this is a reminiscence of how
666 // we used to run metalava separately for API lint and the "last_released" check. Unify them.
667 if doApiLint {
668 rule.Command().Text("touch").Output(d.apiLintTimestamp)
669 }
670 if doCheckReleased {
671 rule.Command().Text("touch").Output(d.checkLastReleasedApiTimestamp)
672 }
673
Colin Cross6aa5c402021-03-24 12:28:50 -0700674 // TODO(b/183630617): rewrapper doesn't support restat rules
675 if !sandbox {
676 rule.Restat()
677 }
Colin Cross2207f872021-03-24 12:39:08 -0700678
679 zipSyncCleanupCmd(rule, srcJarDir)
680
681 rule.Build("metalava", "metalava merged")
682
683 if apiCheckEnabled(ctx, d.properties.Check_api.Current, "current") {
684
685 if len(d.Javadoc.properties.Out) > 0 {
686 ctx.PropertyErrorf("out", "out property may not be combined with check_api")
687 }
688
689 apiFile := android.PathForModuleSrc(ctx, String(d.properties.Check_api.Current.Api_file))
690 removedApiFile := android.PathForModuleSrc(ctx, String(d.properties.Check_api.Current.Removed_api_file))
691 baselineFile := android.OptionalPathForModuleSrc(ctx, d.properties.Check_api.Current.Baseline_file)
692
693 if baselineFile.Valid() {
694 ctx.PropertyErrorf("baseline_file", "current API check can't have a baseline file. (module %s)", ctx.ModuleName())
695 }
696
Colin Crosscb77f752021-03-24 12:04:44 -0700697 d.checkCurrentApiTimestamp = android.PathForModuleOut(ctx, "metalava", "check_current_api.timestamp")
Colin Cross2207f872021-03-24 12:39:08 -0700698
699 rule := android.NewRuleBuilder(pctx, ctx)
700
701 // Diff command line.
702 // -F matches the closest "opening" line, such as "package android {"
703 // and " public class Intent {".
704 diff := `diff -u -F '{ *$'`
705
706 rule.Command().Text("( true")
707 rule.Command().
708 Text(diff).
709 Input(apiFile).Input(d.apiFile)
710
711 rule.Command().
712 Text(diff).
713 Input(removedApiFile).Input(d.removedApiFile)
714
715 msg := fmt.Sprintf(`\n******************************\n`+
716 `You have tried to change the API from what has been previously approved.\n\n`+
717 `To make these errors go away, you have two choices:\n`+
718 ` 1. You can add '@hide' javadoc comments (and remove @SystemApi/@TestApi/etc)\n`+
719 ` to the new methods, etc. shown in the above diff.\n\n`+
720 ` 2. You can update current.txt and/or removed.txt by executing the following command:\n`+
721 ` m %s-update-current-api\n\n`+
722 ` To submit the revised current.txt to the main Android repository,\n`+
723 ` you will need approval.\n`+
724 `******************************\n`, ctx.ModuleName())
725
726 rule.Command().
727 Text("touch").Output(d.checkCurrentApiTimestamp).
728 Text(") || (").
729 Text("echo").Flag("-e").Flag(`"` + msg + `"`).
730 Text("; exit 38").
731 Text(")")
732
733 rule.Build("metalavaCurrentApiCheck", "check current API")
734
Colin Crosscb77f752021-03-24 12:04:44 -0700735 d.updateCurrentApiTimestamp = android.PathForModuleOut(ctx, "metalava", "update_current_api.timestamp")
Colin Cross2207f872021-03-24 12:39:08 -0700736
737 // update API rule
738 rule = android.NewRuleBuilder(pctx, ctx)
739
740 rule.Command().Text("( true")
741
742 rule.Command().
743 Text("cp").Flag("-f").
744 Input(d.apiFile).Flag(apiFile.String())
745
746 rule.Command().
747 Text("cp").Flag("-f").
748 Input(d.removedApiFile).Flag(removedApiFile.String())
749
750 msg = "failed to update public API"
751
752 rule.Command().
753 Text("touch").Output(d.updateCurrentApiTimestamp).
754 Text(") || (").
755 Text("echo").Flag("-e").Flag(`"` + msg + `"`).
756 Text("; exit 38").
757 Text(")")
758
759 rule.Build("metalavaCurrentApiUpdate", "update current API")
760 }
761
762 if String(d.properties.Check_nullability_warnings) != "" {
763 if d.nullabilityWarningsFile == nil {
764 ctx.PropertyErrorf("check_nullability_warnings",
765 "Cannot specify check_nullability_warnings unless validating nullability")
766 }
767
768 checkNullabilityWarnings := android.PathForModuleSrc(ctx, String(d.properties.Check_nullability_warnings))
769
Colin Crosscb77f752021-03-24 12:04:44 -0700770 d.checkNullabilityWarningsTimestamp = android.PathForModuleOut(ctx, "metalava", "check_nullability_warnings.timestamp")
Colin Cross2207f872021-03-24 12:39:08 -0700771
772 msg := fmt.Sprintf(`\n******************************\n`+
773 `The warnings encountered during nullability annotation validation did\n`+
774 `not match the checked in file of expected warnings. The diffs are shown\n`+
775 `above. You have two options:\n`+
776 ` 1. Resolve the differences by editing the nullability annotations.\n`+
777 ` 2. Update the file of expected warnings by running:\n`+
778 ` cp %s %s\n`+
779 ` and submitting the updated file as part of your change.`,
780 d.nullabilityWarningsFile, checkNullabilityWarnings)
781
782 rule := android.NewRuleBuilder(pctx, ctx)
783
784 rule.Command().
785 Text("(").
786 Text("diff").Input(checkNullabilityWarnings).Input(d.nullabilityWarningsFile).
787 Text("&&").
788 Text("touch").Output(d.checkNullabilityWarningsTimestamp).
789 Text(") || (").
790 Text("echo").Flag("-e").Flag(`"` + msg + `"`).
791 Text("; exit 38").
792 Text(")")
793
794 rule.Build("nullabilityWarningsCheck", "nullability warnings check")
795 }
796}
797
798func StubsDefaultsFactory() android.Module {
799 module := &DocDefaults{}
800
801 module.AddProperties(
802 &JavadocProperties{},
803 &DroidstubsProperties{},
804 )
805
806 android.InitDefaultsModule(module)
807
808 return module
809}
810
811var _ android.PrebuiltInterface = (*PrebuiltStubsSources)(nil)
812
813type PrebuiltStubsSourcesProperties struct {
814 Srcs []string `android:"path"`
815}
816
817type PrebuiltStubsSources struct {
818 android.ModuleBase
819 android.DefaultableModuleBase
820 prebuilt android.Prebuilt
821 android.SdkBase
822
823 properties PrebuiltStubsSourcesProperties
824
825 stubsSrcJar android.ModuleOutPath
826}
827
828func (p *PrebuiltStubsSources) OutputFiles(tag string) (android.Paths, error) {
829 switch tag {
830 case "":
831 return android.Paths{p.stubsSrcJar}, nil
832 default:
833 return nil, fmt.Errorf("unsupported module reference tag %q", tag)
834 }
835}
836
837func (d *PrebuiltStubsSources) StubsSrcJar() android.Path {
838 return d.stubsSrcJar
839}
840
841func (p *PrebuiltStubsSources) GenerateAndroidBuildActions(ctx android.ModuleContext) {
842 p.stubsSrcJar = android.PathForModuleOut(ctx, ctx.ModuleName()+"-"+"stubs.srcjar")
843
844 if len(p.properties.Srcs) != 1 {
845 ctx.PropertyErrorf("srcs", "must only specify one directory path, contains %d paths", len(p.properties.Srcs))
846 return
847 }
848
849 localSrcDir := p.properties.Srcs[0]
850 // Although PathForModuleSrc can return nil if either the path doesn't exist or
851 // the path components are invalid it won't in this case because no components
852 // are specified and the module directory must exist in order to get this far.
853 srcDir := android.PathForModuleSrc(ctx).(android.SourcePath).Join(ctx, localSrcDir)
854
855 // Glob the contents of the directory just in case the directory does not exist.
856 srcGlob := localSrcDir + "/**/*"
857 srcPaths := android.PathsForModuleSrc(ctx, []string{srcGlob})
858
859 rule := android.NewRuleBuilder(pctx, ctx)
860 rule.Command().
861 BuiltTool("soong_zip").
862 Flag("-write_if_changed").
863 Flag("-jar").
864 FlagWithOutput("-o ", p.stubsSrcJar).
865 FlagWithArg("-C ", srcDir.String()).
866 FlagWithRspFileInputList("-r ", p.stubsSrcJar.ReplaceExtension(ctx, "rsp"), srcPaths)
867
868 rule.Restat()
869
870 rule.Build("zip src", "Create srcjar from prebuilt source")
871}
872
873func (p *PrebuiltStubsSources) Prebuilt() *android.Prebuilt {
874 return &p.prebuilt
875}
876
877func (p *PrebuiltStubsSources) Name() string {
878 return p.prebuilt.Name(p.ModuleBase.Name())
879}
880
881// prebuilt_stubs_sources imports a set of java source files as if they were
882// generated by droidstubs.
883//
884// By default, a prebuilt_stubs_sources has a single variant that expects a
885// set of `.java` files generated by droidstubs.
886//
887// Specifying `host_supported: true` will produce two variants, one for use as a dependency of device modules and one
888// for host modules.
889//
890// Intended only for use by sdk snapshots.
891func PrebuiltStubsSourcesFactory() android.Module {
892 module := &PrebuiltStubsSources{}
893
894 module.AddProperties(&module.properties)
895
896 android.InitPrebuiltModule(module, &module.properties.Srcs)
897 android.InitSdkAwareModule(module)
898 InitDroiddocModule(module, android.HostAndDeviceSupported)
899 return module
900}