blob: 2b5a21d8979f161ca6945a7f63426a7cbb245037 [file] [log] [blame]
Colin Crossfb6d7812019-01-09 22:17:55 -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 (
Colin Crossfb6d7812019-01-09 22:17:55 -080018 "fmt"
19 "path/filepath"
Colin Cross98fd5742019-01-09 23:04:25 -080020 "sort"
Colin Crossfb6d7812019-01-09 22:17:55 -080021 "strconv"
22 "strings"
Colin Cross3047fa22019-04-18 10:56:44 -070023
Jaewoong Jung9befb0c2020-01-18 10:33:43 -080024 "android/soong/android"
25 "android/soong/java/config"
26
Colin Cross3047fa22019-04-18 10:56:44 -070027 "github.com/google/blueprint/pathtools"
Colin Crossfb6d7812019-01-09 22:17:55 -080028)
29
Colin Cross98fd5742019-01-09 23:04:25 -080030func init() {
Colin Cross3047fa22019-04-18 10:56:44 -070031 android.RegisterPreSingletonType("sdk_versions", sdkPreSingletonFactory)
32 android.RegisterSingletonType("sdk", sdkSingletonFactory)
Colin Cross10932872019-04-18 14:27:12 -070033 android.RegisterMakeVarsProvider(pctx, sdkMakeVars)
Colin Cross98fd5742019-01-09 23:04:25 -080034}
35
Colin Cross3047fa22019-04-18 10:56:44 -070036var sdkVersionsKey = android.NewOnceKey("sdkVersionsKey")
37var sdkFrameworkAidlPathKey = android.NewOnceKey("sdkFrameworkAidlPathKey")
Colin Cross10932872019-04-18 14:27:12 -070038var apiFingerprintPathKey = android.NewOnceKey("apiFingerprintPathKey")
Colin Cross98fd5742019-01-09 23:04:25 -080039
Colin Crossfb6d7812019-01-09 22:17:55 -080040type sdkContext interface {
Jiyong Park6a927c42020-01-21 02:03:43 +090041 // sdkVersion returns sdkSpec that corresponds to the sdk_version property of the current module
42 sdkVersion() sdkSpec
Paul Duffine25c6442019-10-11 13:50:28 +010043 // systemModules returns the system_modules property of the current module, or an empty string if it is not set.
44 systemModules() string
Jiyong Park6a927c42020-01-21 02:03:43 +090045 // minSdkVersion returns sdkSpec that corresponds to the min_sdk_version property of the current module,
46 // or from sdk_version if it is not set.
47 minSdkVersion() sdkSpec
48 // targetSdkVersion returns the sdkSpec that corresponds to the target_sdk_version property of the current module,
49 // or from sdk_version if it is not set.
50 targetSdkVersion() sdkSpec
Colin Crossfb6d7812019-01-09 22:17:55 -080051}
52
Nikita Ioffe934c4f22020-03-02 16:58:11 +000053func UseApiFingerprint(ctx android.BaseModuleContext) bool {
54 if ctx.Config().UnbundledBuild() &&
Baligh Uddinf6201372020-01-24 23:15:44 +000055 !ctx.Config().UnbundledBuildUsePrebuiltSdks() &&
56 ctx.Config().IsEnvTrue("UNBUNDLED_BUILD_TARGET_SDK_WITH_API_FINGERPRINT") {
57 return true
58 }
59 return false
60}
61
Jiyong Park6a927c42020-01-21 02:03:43 +090062// sdkKind represents a particular category of an SDK spec like public, system, test, etc.
63type sdkKind int
64
65const (
66 sdkInvalid sdkKind = iota
67 sdkNone
68 sdkCore
69 sdkCorePlatform
70 sdkPublic
71 sdkSystem
72 sdkTest
Jiyong Park50146e92020-01-30 18:00:15 +090073 sdkModule
Jiyong Parkaae9bd12020-02-12 04:36:43 +090074 sdkSystemServer
Jiyong Park6a927c42020-01-21 02:03:43 +090075 sdkPrivate
76)
77
78// String returns the string representation of this sdkKind
79func (k sdkKind) String() string {
80 switch k {
81 case sdkPrivate:
82 return "private"
83 case sdkNone:
84 return "none"
85 case sdkPublic:
86 return "public"
87 case sdkSystem:
88 return "system"
89 case sdkTest:
90 return "test"
91 case sdkCore:
92 return "core"
93 case sdkCorePlatform:
94 return "core_platform"
Jiyong Park50146e92020-01-30 18:00:15 +090095 case sdkModule:
96 return "module"
Jiyong Parkaae9bd12020-02-12 04:36:43 +090097 case sdkSystemServer:
98 return "system_server"
Colin Crossfb6d7812019-01-09 22:17:55 -080099 default:
Jiyong Park6a927c42020-01-21 02:03:43 +0900100 return "invalid"
Colin Crossfb6d7812019-01-09 22:17:55 -0800101 }
102}
103
Jiyong Park6a927c42020-01-21 02:03:43 +0900104// sdkVersion represents a specific version number of an SDK spec of a particular kind
105type sdkVersion int
106
107const (
108 // special version number for a not-yet-frozen SDK
109 sdkVersionCurrent sdkVersion = sdkVersion(android.FutureApiLevel)
110 // special version number to be used for SDK specs where version number doesn't
111 // make sense, e.g. "none", "", etc.
112 sdkVersionNone sdkVersion = sdkVersion(0)
113)
114
115// isCurrent checks if the sdkVersion refers to the not-yet-published version of an sdkKind
116func (v sdkVersion) isCurrent() bool {
117 return v == sdkVersionCurrent
118}
119
120// isNumbered checks if the sdkVersion refers to the published (a.k.a numbered) version of an sdkKind
121func (v sdkVersion) isNumbered() bool {
122 return !v.isCurrent() && v != sdkVersionNone
123}
124
125// String returns the string representation of this sdkVersion.
126func (v sdkVersion) String() string {
127 if v.isCurrent() {
128 return "current"
129 } else if v.isNumbered() {
130 return strconv.Itoa(int(v))
131 }
132 return "(no version)"
133}
134
135// asNumberString directly converts the numeric value of this sdk version as a string.
136// When isNumbered() is true, this method is the same as String(). However, for sdkVersionCurrent
137// and sdkVersionNone, this returns 10000 and 0 while String() returns "current" and "(no version"),
138// respectively.
139func (v sdkVersion) asNumberString() string {
140 return strconv.Itoa(int(v))
141}
142
143// sdkSpec represents the kind and the version of an SDK for a module to build against
144type sdkSpec struct {
145 kind sdkKind
146 version sdkVersion
147 raw string
148}
149
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100150func (s sdkSpec) String() string {
151 return fmt.Sprintf("%s_%s", s.kind, s.version)
152}
153
Jiyong Park6a927c42020-01-21 02:03:43 +0900154// valid checks if this sdkSpec is well-formed. Note however that true doesn't mean that the
155// specified SDK actually exists.
156func (s sdkSpec) valid() bool {
157 return s.kind != sdkInvalid
158}
159
160// specified checks if this sdkSpec is well-formed and is not "".
161func (s sdkSpec) specified() bool {
162 return s.valid() && s.kind != sdkPrivate
163}
164
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100165// whether the API surface is managed and versioned, i.e. has .txt file that
166// get frozen on SDK freeze and changes get reviewed by API council.
167func (s sdkSpec) stable() bool {
168 if !s.specified() {
169 return false
170 }
171 switch s.kind {
172 case sdkCore, sdkPublic, sdkSystem, sdkModule, sdkSystemServer:
173 return true
174 case sdkNone, sdkCorePlatform, sdkTest, sdkPrivate:
175 return false
176 default:
177 panic(fmt.Errorf("unknown sdkKind=%v", s.kind))
178 }
179 return false
180}
181
Jiyong Park6a927c42020-01-21 02:03:43 +0900182// prebuiltSdkAvailableForUnbundledBuilt tells whether this sdkSpec can have a prebuilt SDK
183// that can be used for unbundled builds.
184func (s sdkSpec) prebuiltSdkAvailableForUnbundledBuild() bool {
185 // "", "none", and "core_platform" are not available for unbundled build
186 // as we don't/can't have prebuilt stub for the versions
187 return s.kind != sdkPrivate && s.kind != sdkNone && s.kind != sdkCorePlatform
188}
189
190// forPdkBuild converts this sdkSpec into another sdkSpec that is for the PDK builds.
191func (s sdkSpec) forPdkBuild(ctx android.EarlyModuleContext) sdkSpec {
192 // For PDK builds, use the latest SDK version instead of "current" or ""
193 if s.kind == sdkPrivate || s.kind == sdkPublic {
194 kind := s.kind
195 if kind == sdkPrivate {
196 // We don't have prebuilt SDK for private APIs, so use the public SDK
197 // instead. This looks odd, but that's how it has been done.
198 // TODO(b/148271073): investigate the need for this.
199 kind = sdkPublic
Colin Crossfb6d7812019-01-09 22:17:55 -0800200 }
Jiyong Park6a927c42020-01-21 02:03:43 +0900201 version := sdkVersion(LatestSdkVersionInt(ctx))
202 return sdkSpec{kind, version, s.raw}
Colin Crossfb6d7812019-01-09 22:17:55 -0800203 }
Jiyong Park6a927c42020-01-21 02:03:43 +0900204 return s
Colin Crossfb6d7812019-01-09 22:17:55 -0800205}
206
Jiyong Park6a927c42020-01-21 02:03:43 +0900207// usePrebuilt determines whether prebuilt SDK should be used for this sdkSpec with the given context.
208func (s sdkSpec) usePrebuilt(ctx android.EarlyModuleContext) bool {
209 if s.version.isCurrent() {
210 // "current" can be built from source and be from prebuilt SDK
211 return ctx.Config().UnbundledBuildUsePrebuiltSdks()
212 } else if s.version.isNumbered() {
213 // sanity check
214 if s.kind != sdkPublic && s.kind != sdkSystem && s.kind != sdkTest {
215 panic(fmt.Errorf("prebuilt SDK is not not available for sdkKind=%q", s.kind))
216 return false
217 }
218 // numbered SDKs are always from prebuilt
219 return true
Colin Crossfb6d7812019-01-09 22:17:55 -0800220 }
Jiyong Park6a927c42020-01-21 02:03:43 +0900221 // "", "none", "core_platform" fall here
222 return false
223}
224
225// effectiveVersion converts an sdkSpec into the concrete sdkVersion that the module
226// should use. For modules targeting an unreleased SDK (meaning it does not yet have a number)
227// it returns android.FutureApiLevel(10000).
228func (s sdkSpec) effectiveVersion(ctx android.EarlyModuleContext) (sdkVersion, error) {
229 if !s.valid() {
230 return s.version, fmt.Errorf("invalid sdk version %q", s.raw)
231 }
232 if ctx.Config().IsPdkBuild() {
233 s = s.forPdkBuild(ctx)
234 }
235 if s.version.isNumbered() {
236 return s.version, nil
237 }
238 return sdkVersion(ctx.Config().DefaultAppTargetSdkInt()), nil
239}
240
241// effectiveVersionString converts an sdkSpec into the concrete version string that the module
242// should use. For modules targeting an unreleased SDK (meaning it does not yet have a number)
243// it returns the codename (P, Q, R, etc.)
244func (s sdkSpec) effectiveVersionString(ctx android.EarlyModuleContext) (string, error) {
245 ver, err := s.effectiveVersion(ctx)
246 if err == nil && int(ver) == ctx.Config().DefaultAppTargetSdkInt() {
247 return ctx.Config().DefaultAppTargetSdk(), nil
248 }
249 return ver.String(), err
250}
251
252func sdkSpecFrom(str string) sdkSpec {
253 switch str {
254 // special cases first
255 case "":
256 return sdkSpec{sdkPrivate, sdkVersionNone, str}
257 case "none":
258 return sdkSpec{sdkNone, sdkVersionNone, str}
259 case "core_platform":
260 return sdkSpec{sdkCorePlatform, sdkVersionNone, str}
261 default:
262 // the syntax is [kind_]version
263 sep := strings.LastIndex(str, "_")
264
265 var kindString string
266 if sep == 0 {
267 return sdkSpec{sdkInvalid, sdkVersionNone, str}
268 } else if sep == -1 {
269 kindString = ""
270 } else {
271 kindString = str[0:sep]
272 }
273 versionString := str[sep+1 : len(str)]
274
275 var kind sdkKind
276 switch kindString {
277 case "":
278 kind = sdkPublic
279 case "core":
280 kind = sdkCore
281 case "system":
282 kind = sdkSystem
283 case "test":
284 kind = sdkTest
Jiyong Park50146e92020-01-30 18:00:15 +0900285 case "module":
286 kind = sdkModule
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900287 case "system_server":
288 kind = sdkSystemServer
Jiyong Park6a927c42020-01-21 02:03:43 +0900289 default:
290 return sdkSpec{sdkInvalid, sdkVersionNone, str}
291 }
292
293 var version sdkVersion
294 if versionString == "current" {
295 version = sdkVersionCurrent
296 } else if i, err := strconv.Atoi(versionString); err == nil {
297 version = sdkVersion(i)
298 } else {
299 return sdkSpec{sdkInvalid, sdkVersionNone, str}
300 }
301
302 return sdkSpec{kind, version, str}
303 }
Colin Crossfb6d7812019-01-09 22:17:55 -0800304}
305
Colin Cross1184b642019-12-30 18:43:07 -0800306func decodeSdkDep(ctx android.EarlyModuleContext, sdkContext sdkContext) sdkDep {
Jiyong Park6a927c42020-01-21 02:03:43 +0900307 sdkVersion := sdkContext.sdkVersion()
308 if !sdkVersion.valid() {
309 ctx.PropertyErrorf("sdk_version", "invalid version %q", sdkVersion.raw)
Colin Crossfb6d7812019-01-09 22:17:55 -0800310 return sdkDep{}
311 }
312
Jiyong Park6a927c42020-01-21 02:03:43 +0900313 if ctx.Config().IsPdkBuild() {
314 sdkVersion = sdkVersion.forPdkBuild(ctx)
315 }
316
317 if sdkVersion.usePrebuilt(ctx) {
318 dir := filepath.Join("prebuilts", "sdk", sdkVersion.version.String(), sdkVersion.kind.String())
Colin Crossfb6d7812019-01-09 22:17:55 -0800319 jar := filepath.Join(dir, "android.jar")
320 // There's no aidl for other SDKs yet.
321 // TODO(77525052): Add aidl files for other SDKs too.
Jiyong Park6a927c42020-01-21 02:03:43 +0900322 public_dir := filepath.Join("prebuilts", "sdk", sdkVersion.version.String(), "public")
Colin Crossfb6d7812019-01-09 22:17:55 -0800323 aidl := filepath.Join(public_dir, "framework.aidl")
324 jarPath := android.ExistentPathForSource(ctx, jar)
325 aidlPath := android.ExistentPathForSource(ctx, aidl)
326 lambdaStubsPath := android.PathForSource(ctx, config.SdkLambdaStubsPath)
327
328 if (!jarPath.Valid() || !aidlPath.Valid()) && ctx.Config().AllowMissingDependencies() {
329 return sdkDep{
330 invalidVersion: true,
Jiyong Park6a927c42020-01-21 02:03:43 +0900331 bootclasspath: []string{fmt.Sprintf("sdk_%s_%s_android", sdkVersion.kind, sdkVersion.version.String())},
Colin Crossfb6d7812019-01-09 22:17:55 -0800332 }
333 }
334
335 if !jarPath.Valid() {
Jiyong Park6a927c42020-01-21 02:03:43 +0900336 ctx.PropertyErrorf("sdk_version", "invalid sdk version %q, %q does not exist", sdkVersion.raw, jar)
Colin Crossfb6d7812019-01-09 22:17:55 -0800337 return sdkDep{}
338 }
339
340 if !aidlPath.Valid() {
Jiyong Park6a927c42020-01-21 02:03:43 +0900341 ctx.PropertyErrorf("sdk_version", "invalid sdk version %q, %q does not exist", sdkVersion.raw, aidl)
Colin Crossfb6d7812019-01-09 22:17:55 -0800342 return sdkDep{}
343 }
344
345 return sdkDep{
346 useFiles: true,
347 jars: android.Paths{jarPath.Path(), lambdaStubsPath},
Colin Cross3047fa22019-04-18 10:56:44 -0700348 aidl: android.OptionalPathForPath(aidlPath.Path()),
Colin Crossfb6d7812019-01-09 22:17:55 -0800349 }
350 }
351
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900352 toModule := func(modules []string, res string, aidl android.Path) sdkDep {
Colin Cross6cef4812019-10-17 14:23:50 -0700353 return sdkDep{
Colin Crossfb6d7812019-01-09 22:17:55 -0800354 useModule: true,
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900355 bootclasspath: append(modules, config.DefaultLambdaStubsLibrary),
Colin Cross6cef4812019-10-17 14:23:50 -0700356 systemModules: "core-current-stubs-system-modules",
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900357 java9Classpath: modules,
358 frameworkResModule: res,
Colin Cross3047fa22019-04-18 10:56:44 -0700359 aidl: android.OptionalPathForPath(aidl),
Colin Crossfb6d7812019-01-09 22:17:55 -0800360 }
Colin Crossfb6d7812019-01-09 22:17:55 -0800361 }
362
Colin Cross98fd5742019-01-09 23:04:25 -0800363 // Ensures that the specificed system SDK version is one of BOARD_SYSTEMSDK_VERSIONS (for vendor apks)
364 // or PRODUCT_SYSTEMSDK_VERSIONS (for other apks or when BOARD_SYSTEMSDK_VERSIONS is not set)
Jiyong Park6a927c42020-01-21 02:03:43 +0900365 if sdkVersion.kind == sdkSystem && sdkVersion.version.isNumbered() {
Colin Cross98fd5742019-01-09 23:04:25 -0800366 allowed_versions := ctx.DeviceConfig().PlatformSystemSdkVersions()
367 if ctx.DeviceSpecific() || ctx.SocSpecific() {
368 if len(ctx.DeviceConfig().SystemSdkVersions()) > 0 {
369 allowed_versions = ctx.DeviceConfig().SystemSdkVersions()
370 }
371 }
Jiyong Park6a927c42020-01-21 02:03:43 +0900372 if len(allowed_versions) > 0 && !android.InList(sdkVersion.version.String(), allowed_versions) {
Colin Cross98fd5742019-01-09 23:04:25 -0800373 ctx.PropertyErrorf("sdk_version", "incompatible sdk version %q. System SDK version should be one of %q",
Jiyong Park6a927c42020-01-21 02:03:43 +0900374 sdkVersion.raw, allowed_versions)
Colin Cross98fd5742019-01-09 23:04:25 -0800375 }
376 }
377
Jiyong Park6a927c42020-01-21 02:03:43 +0900378 switch sdkVersion.kind {
379 case sdkPrivate:
Colin Crossfb6d7812019-01-09 22:17:55 -0800380 return sdkDep{
381 useDefaultLibs: true,
382 frameworkResModule: "framework-res",
383 }
Jiyong Park6a927c42020-01-21 02:03:43 +0900384 case sdkNone:
Paul Duffine25c6442019-10-11 13:50:28 +0100385 systemModules := sdkContext.systemModules()
386 if systemModules == "" {
387 ctx.PropertyErrorf("sdk_version",
388 `system_modules is required to be set to a non-empty value when sdk_version is "none", did you mean sdk_version: "core_platform"?`)
389 } else if systemModules == "none" {
Colin Cross6d8d8c62019-10-28 15:10:03 -0700390 return sdkDep{
391 noStandardLibs: true,
392 }
Paul Duffine25c6442019-10-11 13:50:28 +0100393 }
394
Paul Duffin52d398a2019-06-11 12:31:14 +0100395 return sdkDep{
Colin Cross6d8d8c62019-10-28 15:10:03 -0700396 useModule: true,
Paul Duffin52d398a2019-06-11 12:31:14 +0100397 noStandardLibs: true,
Paul Duffine25c6442019-10-11 13:50:28 +0100398 systemModules: systemModules,
Colin Cross6cef4812019-10-17 14:23:50 -0700399 bootclasspath: []string{systemModules},
Paul Duffin52d398a2019-06-11 12:31:14 +0100400 }
Jiyong Park6a927c42020-01-21 02:03:43 +0900401 case sdkCorePlatform:
Paul Duffin50c217c2019-06-12 13:25:22 +0100402 return sdkDep{
403 useDefaultLibs: true,
404 frameworkResModule: "framework-res",
405 noFrameworksLibs: true,
406 }
Jiyong Park6a927c42020-01-21 02:03:43 +0900407 case sdkPublic:
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900408 return toModule([]string{"android_stubs_current"}, "framework-res", sdkFrameworkAidlPath(ctx))
Jiyong Park6a927c42020-01-21 02:03:43 +0900409 case sdkSystem:
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900410 return toModule([]string{"android_system_stubs_current"}, "framework-res", sdkFrameworkAidlPath(ctx))
Jiyong Park6a927c42020-01-21 02:03:43 +0900411 case sdkTest:
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900412 return toModule([]string{"android_test_stubs_current"}, "framework-res", sdkFrameworkAidlPath(ctx))
Jiyong Park6a927c42020-01-21 02:03:43 +0900413 case sdkCore:
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900414 return toModule([]string{"core.current.stubs"}, "", nil)
Jiyong Park50146e92020-01-30 18:00:15 +0900415 case sdkModule:
416 // TODO(146757305): provide .apk and .aidl that have more APIs for modules
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900417 return toModule([]string{"android_module_lib_stubs_current"}, "framework-res", sdkFrameworkAidlPath(ctx))
418 case sdkSystemServer:
419 // TODO(146757305): provide .apk and .aidl that have more APIs for modules
Anton Hanssonbbd78552020-03-19 15:23:38 +0000420 return toModule([]string{"android_system_server_stubs_current"}, "framework-res", sdkFrameworkAidlPath(ctx))
Colin Crossfb6d7812019-01-09 22:17:55 -0800421 default:
Jiyong Park6a927c42020-01-21 02:03:43 +0900422 panic(fmt.Errorf("invalid sdk %q", sdkVersion.raw))
Colin Crossfb6d7812019-01-09 22:17:55 -0800423 }
424}
Colin Cross98fd5742019-01-09 23:04:25 -0800425
Colin Cross3047fa22019-04-18 10:56:44 -0700426func sdkPreSingletonFactory() android.Singleton {
427 return sdkPreSingleton{}
Colin Cross98fd5742019-01-09 23:04:25 -0800428}
429
Colin Cross3047fa22019-04-18 10:56:44 -0700430type sdkPreSingleton struct{}
Colin Cross98fd5742019-01-09 23:04:25 -0800431
Colin Cross3047fa22019-04-18 10:56:44 -0700432func (sdkPreSingleton) GenerateBuildActions(ctx android.SingletonContext) {
Colin Cross98fd5742019-01-09 23:04:25 -0800433 sdkJars, err := ctx.GlobWithDeps("prebuilts/sdk/*/public/android.jar", nil)
434 if err != nil {
435 ctx.Errorf("failed to glob prebuilts/sdk/*/public/android.jar: %s", err.Error())
436 }
437
438 var sdkVersions []int
439 for _, sdkJar := range sdkJars {
440 dir := filepath.Base(filepath.Dir(filepath.Dir(sdkJar)))
441 v, err := strconv.Atoi(dir)
442 if scerr, ok := err.(*strconv.NumError); ok && scerr.Err == strconv.ErrSyntax {
443 continue
444 } else if err != nil {
445 ctx.Errorf("invalid sdk jar %q, %s, %v", sdkJar, err.Error())
446 }
447 sdkVersions = append(sdkVersions, v)
448 }
449
450 sort.Ints(sdkVersions)
451
Colin Cross3047fa22019-04-18 10:56:44 -0700452 ctx.Config().Once(sdkVersionsKey, func() interface{} { return sdkVersions })
453}
454
Jiyong Park6a927c42020-01-21 02:03:43 +0900455func LatestSdkVersionInt(ctx android.EarlyModuleContext) int {
456 sdkVersions := ctx.Config().Get(sdkVersionsKey).([]int)
457 latestSdkVersion := 0
458 if len(sdkVersions) > 0 {
459 latestSdkVersion = sdkVersions[len(sdkVersions)-1]
460 }
461 return latestSdkVersion
462}
463
Colin Cross3047fa22019-04-18 10:56:44 -0700464func sdkSingletonFactory() android.Singleton {
465 return sdkSingleton{}
466}
467
468type sdkSingleton struct{}
469
470func (sdkSingleton) GenerateBuildActions(ctx android.SingletonContext) {
Colin Cross10932872019-04-18 14:27:12 -0700471 if ctx.Config().UnbundledBuildUsePrebuiltSdks() || ctx.Config().IsPdkBuild() {
Colin Cross3047fa22019-04-18 10:56:44 -0700472 return
473 }
474
Colin Cross10932872019-04-18 14:27:12 -0700475 createSdkFrameworkAidl(ctx)
476 createAPIFingerprint(ctx)
477}
Colin Cross3047fa22019-04-18 10:56:44 -0700478
Colin Cross10932872019-04-18 14:27:12 -0700479// Create framework.aidl by extracting anything that implements android.os.Parcelable from the SDK stubs modules.
480func createSdkFrameworkAidl(ctx android.SingletonContext) {
Colin Cross3047fa22019-04-18 10:56:44 -0700481 stubsModules := []string{
482 "android_stubs_current",
483 "android_test_stubs_current",
484 "android_system_stubs_current",
485 }
486
487 stubsJars := make([]android.Paths, len(stubsModules))
488
489 ctx.VisitAllModules(func(module android.Module) {
490 // Collect dex jar paths for the modules listed above.
491 if j, ok := module.(Dependency); ok {
492 name := ctx.ModuleName(module)
493 if i := android.IndexList(name, stubsModules); i != -1 {
494 stubsJars[i] = j.HeaderJars()
495 }
496 }
497 })
498
499 var missingDeps []string
500
501 for i := range stubsJars {
502 if stubsJars[i] == nil {
503 if ctx.Config().AllowMissingDependencies() {
504 missingDeps = append(missingDeps, stubsModules[i])
505 } else {
506 ctx.Errorf("failed to find dex jar path for module %q",
507 stubsModules[i])
508 }
509 }
510 }
511
512 rule := android.NewRuleBuilder()
513 rule.MissingDeps(missingDeps)
514
515 var aidls android.Paths
516 for _, jars := range stubsJars {
517 for _, jar := range jars {
518 aidl := android.PathForOutput(ctx, "aidl", pathtools.ReplaceExtension(jar.Base(), "aidl"))
519
520 rule.Command().
521 Text("rm -f").Output(aidl)
522 rule.Command().
Colin Crossee94d6a2019-07-08 17:08:34 -0700523 BuiltTool(ctx, "sdkparcelables").
Colin Cross3047fa22019-04-18 10:56:44 -0700524 Input(jar).
525 Output(aidl)
526
527 aidls = append(aidls, aidl)
528 }
529 }
530
531 combinedAidl := sdkFrameworkAidlPath(ctx)
532 tempPath := combinedAidl.ReplaceExtension(ctx, "aidl.tmp")
533
534 rule.Command().
535 Text("rm -f").Output(tempPath)
536 rule.Command().
537 Text("cat").
538 Inputs(aidls).
539 Text("| sort -u >").
540 Output(tempPath)
541
542 commitChangeForRestat(rule, tempPath, combinedAidl)
543
544 rule.Build(pctx, ctx, "framework_aidl", "generate framework.aidl")
545}
546
547func sdkFrameworkAidlPath(ctx android.PathContext) android.OutputPath {
548 return ctx.Config().Once(sdkFrameworkAidlPathKey, func() interface{} {
549 return android.PathForOutput(ctx, "framework.aidl")
550 }).(android.OutputPath)
551}
552
Colin Cross10932872019-04-18 14:27:12 -0700553// Create api_fingerprint.txt
554func createAPIFingerprint(ctx android.SingletonContext) {
Jiyong Park71b519d2019-04-18 17:25:49 +0900555 out := ApiFingerprintPath(ctx)
Colin Cross10932872019-04-18 14:27:12 -0700556
557 rule := android.NewRuleBuilder()
558
559 rule.Command().
560 Text("rm -f").Output(out)
561 cmd := rule.Command()
562
563 if ctx.Config().PlatformSdkCodename() == "REL" {
564 cmd.Text("echo REL >").Output(out)
565 } else if ctx.Config().IsPdkBuild() {
566 // TODO: get this from the PDK artifacts?
567 cmd.Text("echo PDK >").Output(out)
568 } else if !ctx.Config().UnbundledBuildUsePrebuiltSdks() {
569 in, err := ctx.GlobWithDeps("frameworks/base/api/*current.txt", nil)
570 if err != nil {
571 ctx.Errorf("error globbing API files: %s", err)
572 }
573
574 cmd.Text("cat").
575 Inputs(android.PathsForSource(ctx, in)).
Elliott Hughes34b49d12019-09-06 14:42:24 -0700576 Text("| md5sum | cut -d' ' -f1 >").
Colin Cross10932872019-04-18 14:27:12 -0700577 Output(out)
578 } else {
579 // Unbundled build
580 // TODO: use a prebuilt api_fingerprint.txt from prebuilts/sdk/current.txt once we have one
581 cmd.Text("echo").
582 Flag(ctx.Config().PlatformPreviewSdkVersion()).
583 Text(">").
584 Output(out)
585 }
586
587 rule.Build(pctx, ctx, "api_fingerprint", "generate api_fingerprint.txt")
588}
589
Jiyong Park71b519d2019-04-18 17:25:49 +0900590func ApiFingerprintPath(ctx android.PathContext) android.OutputPath {
Colin Cross10932872019-04-18 14:27:12 -0700591 return ctx.Config().Once(apiFingerprintPathKey, func() interface{} {
592 return android.PathForOutput(ctx, "api_fingerprint.txt")
593 }).(android.OutputPath)
594}
595
596func sdkMakeVars(ctx android.MakeVarsContext) {
597 if ctx.Config().UnbundledBuildUsePrebuiltSdks() || ctx.Config().IsPdkBuild() {
Colin Cross3047fa22019-04-18 10:56:44 -0700598 return
599 }
600
601 ctx.Strict("FRAMEWORK_AIDL", sdkFrameworkAidlPath(ctx).String())
Jiyong Park71b519d2019-04-18 17:25:49 +0900602 ctx.Strict("API_FINGERPRINT", ApiFingerprintPath(ctx).String())
Colin Cross98fd5742019-01-09 23:04:25 -0800603}