blob: 9310f78460ed554802336b97eb9f62f9fa7ac448 [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 Ioffe1f4f3452020-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 Satayev2db1c3f2020-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 Satayev2db1c3f2020-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 {
Artur Satayev8cf899a2020-04-15 17:29:42 +0100172 case sdkNone:
173 // there is nothing to manage and version in this case; de facto stable API.
174 return true
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100175 case sdkCore, sdkPublic, sdkSystem, sdkModule, sdkSystemServer:
176 return true
Artur Satayev8cf899a2020-04-15 17:29:42 +0100177 case sdkCorePlatform, sdkTest, sdkPrivate:
Artur Satayev2db1c3f2020-04-08 19:09:30 +0100178 return false
179 default:
180 panic(fmt.Errorf("unknown sdkKind=%v", s.kind))
181 }
182 return false
183}
184
Jiyong Park6a927c42020-01-21 02:03:43 +0900185// prebuiltSdkAvailableForUnbundledBuilt tells whether this sdkSpec can have a prebuilt SDK
186// that can be used for unbundled builds.
187func (s sdkSpec) prebuiltSdkAvailableForUnbundledBuild() bool {
188 // "", "none", and "core_platform" are not available for unbundled build
189 // as we don't/can't have prebuilt stub for the versions
190 return s.kind != sdkPrivate && s.kind != sdkNone && s.kind != sdkCorePlatform
191}
192
193// forPdkBuild converts this sdkSpec into another sdkSpec that is for the PDK builds.
194func (s sdkSpec) forPdkBuild(ctx android.EarlyModuleContext) sdkSpec {
195 // For PDK builds, use the latest SDK version instead of "current" or ""
196 if s.kind == sdkPrivate || s.kind == sdkPublic {
197 kind := s.kind
198 if kind == sdkPrivate {
199 // We don't have prebuilt SDK for private APIs, so use the public SDK
200 // instead. This looks odd, but that's how it has been done.
201 // TODO(b/148271073): investigate the need for this.
202 kind = sdkPublic
Colin Crossfb6d7812019-01-09 22:17:55 -0800203 }
Jiyong Park6a927c42020-01-21 02:03:43 +0900204 version := sdkVersion(LatestSdkVersionInt(ctx))
205 return sdkSpec{kind, version, s.raw}
Colin Crossfb6d7812019-01-09 22:17:55 -0800206 }
Jiyong Park6a927c42020-01-21 02:03:43 +0900207 return s
Colin Crossfb6d7812019-01-09 22:17:55 -0800208}
209
Jiyong Park6a927c42020-01-21 02:03:43 +0900210// usePrebuilt determines whether prebuilt SDK should be used for this sdkSpec with the given context.
211func (s sdkSpec) usePrebuilt(ctx android.EarlyModuleContext) bool {
212 if s.version.isCurrent() {
213 // "current" can be built from source and be from prebuilt SDK
214 return ctx.Config().UnbundledBuildUsePrebuiltSdks()
215 } else if s.version.isNumbered() {
216 // sanity check
217 if s.kind != sdkPublic && s.kind != sdkSystem && s.kind != sdkTest {
218 panic(fmt.Errorf("prebuilt SDK is not not available for sdkKind=%q", s.kind))
219 return false
220 }
221 // numbered SDKs are always from prebuilt
222 return true
Colin Crossfb6d7812019-01-09 22:17:55 -0800223 }
Jiyong Park6a927c42020-01-21 02:03:43 +0900224 // "", "none", "core_platform" fall here
225 return false
226}
227
228// effectiveVersion converts an sdkSpec into the concrete sdkVersion that the module
229// should use. For modules targeting an unreleased SDK (meaning it does not yet have a number)
230// it returns android.FutureApiLevel(10000).
231func (s sdkSpec) effectiveVersion(ctx android.EarlyModuleContext) (sdkVersion, error) {
232 if !s.valid() {
233 return s.version, fmt.Errorf("invalid sdk version %q", s.raw)
234 }
235 if ctx.Config().IsPdkBuild() {
236 s = s.forPdkBuild(ctx)
237 }
238 if s.version.isNumbered() {
239 return s.version, nil
240 }
241 return sdkVersion(ctx.Config().DefaultAppTargetSdkInt()), nil
242}
243
244// effectiveVersionString converts an sdkSpec into the concrete version string that the module
245// should use. For modules targeting an unreleased SDK (meaning it does not yet have a number)
246// it returns the codename (P, Q, R, etc.)
247func (s sdkSpec) effectiveVersionString(ctx android.EarlyModuleContext) (string, error) {
248 ver, err := s.effectiveVersion(ctx)
249 if err == nil && int(ver) == ctx.Config().DefaultAppTargetSdkInt() {
250 return ctx.Config().DefaultAppTargetSdk(), nil
251 }
252 return ver.String(), err
253}
254
Colin Cross17dec172020-05-14 18:05:32 -0700255func (s sdkSpec) defaultJavaLanguageVersion(ctx android.EarlyModuleContext) javaVersion {
256 sdk, err := s.effectiveVersion(ctx)
257 if err != nil {
258 ctx.PropertyErrorf("sdk_version", "%s", err)
259 }
260 if sdk <= 23 {
261 return JAVA_VERSION_7
262 } else if sdk <= 29 {
263 return JAVA_VERSION_8
264 } else {
265 return JAVA_VERSION_9
266 }
267}
268
Jiyong Park6a927c42020-01-21 02:03:43 +0900269func sdkSpecFrom(str string) sdkSpec {
270 switch str {
271 // special cases first
272 case "":
273 return sdkSpec{sdkPrivate, sdkVersionNone, str}
274 case "none":
275 return sdkSpec{sdkNone, sdkVersionNone, str}
276 case "core_platform":
277 return sdkSpec{sdkCorePlatform, sdkVersionNone, str}
278 default:
279 // the syntax is [kind_]version
280 sep := strings.LastIndex(str, "_")
281
282 var kindString string
283 if sep == 0 {
284 return sdkSpec{sdkInvalid, sdkVersionNone, str}
285 } else if sep == -1 {
286 kindString = ""
287 } else {
288 kindString = str[0:sep]
289 }
290 versionString := str[sep+1 : len(str)]
291
292 var kind sdkKind
293 switch kindString {
294 case "":
295 kind = sdkPublic
296 case "core":
297 kind = sdkCore
298 case "system":
299 kind = sdkSystem
300 case "test":
301 kind = sdkTest
Jiyong Park50146e92020-01-30 18:00:15 +0900302 case "module":
303 kind = sdkModule
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900304 case "system_server":
305 kind = sdkSystemServer
Jiyong Park6a927c42020-01-21 02:03:43 +0900306 default:
307 return sdkSpec{sdkInvalid, sdkVersionNone, str}
308 }
309
310 var version sdkVersion
311 if versionString == "current" {
312 version = sdkVersionCurrent
313 } else if i, err := strconv.Atoi(versionString); err == nil {
314 version = sdkVersion(i)
315 } else {
316 return sdkSpec{sdkInvalid, sdkVersionNone, str}
317 }
318
319 return sdkSpec{kind, version, str}
320 }
Colin Crossfb6d7812019-01-09 22:17:55 -0800321}
322
Jeongik Cha7c708312020-01-28 13:52:36 +0900323func (s sdkSpec) validateSystemSdk(ctx android.EarlyModuleContext) bool {
324 // Ensures that the specified system SDK version is one of BOARD_SYSTEMSDK_VERSIONS (for vendor/product Java module)
325 // Assuming that BOARD_SYSTEMSDK_VERSIONS := 28 29,
326 // sdk_version of the modules in vendor/product that use system sdk must be either system_28, system_29 or system_current
327 if s.kind != sdkSystem || !s.version.isNumbered() {
328 return true
329 }
330 allowedVersions := ctx.DeviceConfig().PlatformSystemSdkVersions()
331 if ctx.DeviceSpecific() || ctx.SocSpecific() || (ctx.ProductSpecific() && ctx.Config().EnforceProductPartitionInterface()) {
332 systemSdkVersions := ctx.DeviceConfig().SystemSdkVersions()
333 if len(systemSdkVersions) > 0 {
334 allowedVersions = systemSdkVersions
335 }
336 }
337 if len(allowedVersions) > 0 && !android.InList(s.version.String(), allowedVersions) {
338 ctx.PropertyErrorf("sdk_version", "incompatible sdk version %q. System SDK version should be one of %q",
339 s.raw, allowedVersions)
340 return false
341 }
342 return true
343}
344
Colin Cross1184b642019-12-30 18:43:07 -0800345func decodeSdkDep(ctx android.EarlyModuleContext, sdkContext sdkContext) sdkDep {
Jiyong Park6a927c42020-01-21 02:03:43 +0900346 sdkVersion := sdkContext.sdkVersion()
347 if !sdkVersion.valid() {
348 ctx.PropertyErrorf("sdk_version", "invalid version %q", sdkVersion.raw)
Colin Crossfb6d7812019-01-09 22:17:55 -0800349 return sdkDep{}
350 }
351
Jiyong Park6a927c42020-01-21 02:03:43 +0900352 if ctx.Config().IsPdkBuild() {
353 sdkVersion = sdkVersion.forPdkBuild(ctx)
354 }
Jeongik Cha7c708312020-01-28 13:52:36 +0900355 if !sdkVersion.validateSystemSdk(ctx) {
356 return sdkDep{}
357 }
Jiyong Park6a927c42020-01-21 02:03:43 +0900358
359 if sdkVersion.usePrebuilt(ctx) {
360 dir := filepath.Join("prebuilts", "sdk", sdkVersion.version.String(), sdkVersion.kind.String())
Colin Crossfb6d7812019-01-09 22:17:55 -0800361 jar := filepath.Join(dir, "android.jar")
362 // There's no aidl for other SDKs yet.
363 // TODO(77525052): Add aidl files for other SDKs too.
Jiyong Park6a927c42020-01-21 02:03:43 +0900364 public_dir := filepath.Join("prebuilts", "sdk", sdkVersion.version.String(), "public")
Colin Crossfb6d7812019-01-09 22:17:55 -0800365 aidl := filepath.Join(public_dir, "framework.aidl")
366 jarPath := android.ExistentPathForSource(ctx, jar)
367 aidlPath := android.ExistentPathForSource(ctx, aidl)
368 lambdaStubsPath := android.PathForSource(ctx, config.SdkLambdaStubsPath)
369
370 if (!jarPath.Valid() || !aidlPath.Valid()) && ctx.Config().AllowMissingDependencies() {
371 return sdkDep{
372 invalidVersion: true,
Jiyong Park6a927c42020-01-21 02:03:43 +0900373 bootclasspath: []string{fmt.Sprintf("sdk_%s_%s_android", sdkVersion.kind, sdkVersion.version.String())},
Colin Crossfb6d7812019-01-09 22:17:55 -0800374 }
375 }
376
377 if !jarPath.Valid() {
Jiyong Park6a927c42020-01-21 02:03:43 +0900378 ctx.PropertyErrorf("sdk_version", "invalid sdk version %q, %q does not exist", sdkVersion.raw, jar)
Colin Crossfb6d7812019-01-09 22:17:55 -0800379 return sdkDep{}
380 }
381
382 if !aidlPath.Valid() {
Jiyong Park6a927c42020-01-21 02:03:43 +0900383 ctx.PropertyErrorf("sdk_version", "invalid sdk version %q, %q does not exist", sdkVersion.raw, aidl)
Colin Crossfb6d7812019-01-09 22:17:55 -0800384 return sdkDep{}
385 }
386
Colin Cross17dec172020-05-14 18:05:32 -0700387 var systemModules string
388 if sdkVersion.defaultJavaLanguageVersion(ctx).usesJavaModules() {
389 systemModules = "sdk_public_" + sdkVersion.version.String() + "_system_modules"
390 }
391
Colin Crossfb6d7812019-01-09 22:17:55 -0800392 return sdkDep{
Colin Cross17dec172020-05-14 18:05:32 -0700393 useFiles: true,
394 jars: android.Paths{jarPath.Path(), lambdaStubsPath},
395 aidl: android.OptionalPathForPath(aidlPath.Path()),
396 systemModules: systemModules,
Colin Crossfb6d7812019-01-09 22:17:55 -0800397 }
398 }
399
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900400 toModule := func(modules []string, res string, aidl android.Path) sdkDep {
Colin Cross6cef4812019-10-17 14:23:50 -0700401 return sdkDep{
Colin Crossfb6d7812019-01-09 22:17:55 -0800402 useModule: true,
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900403 bootclasspath: append(modules, config.DefaultLambdaStubsLibrary),
Colin Cross6cef4812019-10-17 14:23:50 -0700404 systemModules: "core-current-stubs-system-modules",
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900405 java9Classpath: modules,
406 frameworkResModule: res,
Colin Cross3047fa22019-04-18 10:56:44 -0700407 aidl: android.OptionalPathForPath(aidl),
Colin Crossfb6d7812019-01-09 22:17:55 -0800408 }
Colin Crossfb6d7812019-01-09 22:17:55 -0800409 }
410
Jiyong Park6a927c42020-01-21 02:03:43 +0900411 switch sdkVersion.kind {
412 case sdkPrivate:
Colin Crossfb6d7812019-01-09 22:17:55 -0800413 return sdkDep{
414 useDefaultLibs: true,
415 frameworkResModule: "framework-res",
416 }
Jiyong Park6a927c42020-01-21 02:03:43 +0900417 case sdkNone:
Paul Duffine25c6442019-10-11 13:50:28 +0100418 systemModules := sdkContext.systemModules()
419 if systemModules == "" {
420 ctx.PropertyErrorf("sdk_version",
421 `system_modules is required to be set to a non-empty value when sdk_version is "none", did you mean sdk_version: "core_platform"?`)
422 } else if systemModules == "none" {
Colin Cross6d8d8c62019-10-28 15:10:03 -0700423 return sdkDep{
424 noStandardLibs: true,
425 }
Paul Duffine25c6442019-10-11 13:50:28 +0100426 }
427
Paul Duffin52d398a2019-06-11 12:31:14 +0100428 return sdkDep{
Colin Cross6d8d8c62019-10-28 15:10:03 -0700429 useModule: true,
Paul Duffin52d398a2019-06-11 12:31:14 +0100430 noStandardLibs: true,
Paul Duffine25c6442019-10-11 13:50:28 +0100431 systemModules: systemModules,
Colin Cross6cef4812019-10-17 14:23:50 -0700432 bootclasspath: []string{systemModules},
Paul Duffin52d398a2019-06-11 12:31:14 +0100433 }
Jiyong Park6a927c42020-01-21 02:03:43 +0900434 case sdkCorePlatform:
Paul Duffin50c217c2019-06-12 13:25:22 +0100435 return sdkDep{
436 useDefaultLibs: true,
437 frameworkResModule: "framework-res",
438 noFrameworksLibs: true,
439 }
Jiyong Park6a927c42020-01-21 02:03:43 +0900440 case sdkPublic:
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900441 return toModule([]string{"android_stubs_current"}, "framework-res", sdkFrameworkAidlPath(ctx))
Jiyong Park6a927c42020-01-21 02:03:43 +0900442 case sdkSystem:
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900443 return toModule([]string{"android_system_stubs_current"}, "framework-res", sdkFrameworkAidlPath(ctx))
Jiyong Park6a927c42020-01-21 02:03:43 +0900444 case sdkTest:
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900445 return toModule([]string{"android_test_stubs_current"}, "framework-res", sdkFrameworkAidlPath(ctx))
Jiyong Park6a927c42020-01-21 02:03:43 +0900446 case sdkCore:
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900447 return toModule([]string{"core.current.stubs"}, "", nil)
Jiyong Park50146e92020-01-30 18:00:15 +0900448 case sdkModule:
449 // TODO(146757305): provide .apk and .aidl that have more APIs for modules
Jiyong Parkaae9bd12020-02-12 04:36:43 +0900450 return toModule([]string{"android_module_lib_stubs_current"}, "framework-res", sdkFrameworkAidlPath(ctx))
451 case sdkSystemServer:
452 // TODO(146757305): provide .apk and .aidl that have more APIs for modules
Anton Hanssonba6ab2e2020-03-19 15:23:38 +0000453 return toModule([]string{"android_system_server_stubs_current"}, "framework-res", sdkFrameworkAidlPath(ctx))
Colin Crossfb6d7812019-01-09 22:17:55 -0800454 default:
Jiyong Park6a927c42020-01-21 02:03:43 +0900455 panic(fmt.Errorf("invalid sdk %q", sdkVersion.raw))
Colin Crossfb6d7812019-01-09 22:17:55 -0800456 }
457}
Colin Cross98fd5742019-01-09 23:04:25 -0800458
Colin Cross3047fa22019-04-18 10:56:44 -0700459func sdkPreSingletonFactory() android.Singleton {
460 return sdkPreSingleton{}
Colin Cross98fd5742019-01-09 23:04:25 -0800461}
462
Colin Cross3047fa22019-04-18 10:56:44 -0700463type sdkPreSingleton struct{}
Colin Cross98fd5742019-01-09 23:04:25 -0800464
Colin Cross3047fa22019-04-18 10:56:44 -0700465func (sdkPreSingleton) GenerateBuildActions(ctx android.SingletonContext) {
Colin Cross98fd5742019-01-09 23:04:25 -0800466 sdkJars, err := ctx.GlobWithDeps("prebuilts/sdk/*/public/android.jar", nil)
467 if err != nil {
468 ctx.Errorf("failed to glob prebuilts/sdk/*/public/android.jar: %s", err.Error())
469 }
470
471 var sdkVersions []int
472 for _, sdkJar := range sdkJars {
473 dir := filepath.Base(filepath.Dir(filepath.Dir(sdkJar)))
474 v, err := strconv.Atoi(dir)
475 if scerr, ok := err.(*strconv.NumError); ok && scerr.Err == strconv.ErrSyntax {
476 continue
477 } else if err != nil {
478 ctx.Errorf("invalid sdk jar %q, %s, %v", sdkJar, err.Error())
479 }
480 sdkVersions = append(sdkVersions, v)
481 }
482
483 sort.Ints(sdkVersions)
484
Colin Cross3047fa22019-04-18 10:56:44 -0700485 ctx.Config().Once(sdkVersionsKey, func() interface{} { return sdkVersions })
486}
487
Jiyong Park6a927c42020-01-21 02:03:43 +0900488func LatestSdkVersionInt(ctx android.EarlyModuleContext) int {
489 sdkVersions := ctx.Config().Get(sdkVersionsKey).([]int)
490 latestSdkVersion := 0
491 if len(sdkVersions) > 0 {
492 latestSdkVersion = sdkVersions[len(sdkVersions)-1]
493 }
494 return latestSdkVersion
495}
496
Colin Cross3047fa22019-04-18 10:56:44 -0700497func sdkSingletonFactory() android.Singleton {
498 return sdkSingleton{}
499}
500
501type sdkSingleton struct{}
502
503func (sdkSingleton) GenerateBuildActions(ctx android.SingletonContext) {
Colin Cross10932872019-04-18 14:27:12 -0700504 if ctx.Config().UnbundledBuildUsePrebuiltSdks() || ctx.Config().IsPdkBuild() {
Colin Cross3047fa22019-04-18 10:56:44 -0700505 return
506 }
507
Colin Cross10932872019-04-18 14:27:12 -0700508 createSdkFrameworkAidl(ctx)
509 createAPIFingerprint(ctx)
510}
Colin Cross3047fa22019-04-18 10:56:44 -0700511
Colin Cross10932872019-04-18 14:27:12 -0700512// Create framework.aidl by extracting anything that implements android.os.Parcelable from the SDK stubs modules.
513func createSdkFrameworkAidl(ctx android.SingletonContext) {
Colin Cross3047fa22019-04-18 10:56:44 -0700514 stubsModules := []string{
515 "android_stubs_current",
516 "android_test_stubs_current",
517 "android_system_stubs_current",
518 }
519
520 stubsJars := make([]android.Paths, len(stubsModules))
521
522 ctx.VisitAllModules(func(module android.Module) {
523 // Collect dex jar paths for the modules listed above.
524 if j, ok := module.(Dependency); ok {
525 name := ctx.ModuleName(module)
526 if i := android.IndexList(name, stubsModules); i != -1 {
527 stubsJars[i] = j.HeaderJars()
528 }
529 }
530 })
531
532 var missingDeps []string
533
534 for i := range stubsJars {
535 if stubsJars[i] == nil {
536 if ctx.Config().AllowMissingDependencies() {
537 missingDeps = append(missingDeps, stubsModules[i])
538 } else {
539 ctx.Errorf("failed to find dex jar path for module %q",
540 stubsModules[i])
541 }
542 }
543 }
544
545 rule := android.NewRuleBuilder()
546 rule.MissingDeps(missingDeps)
547
548 var aidls android.Paths
549 for _, jars := range stubsJars {
550 for _, jar := range jars {
551 aidl := android.PathForOutput(ctx, "aidl", pathtools.ReplaceExtension(jar.Base(), "aidl"))
552
553 rule.Command().
554 Text("rm -f").Output(aidl)
555 rule.Command().
Colin Crossee94d6a2019-07-08 17:08:34 -0700556 BuiltTool(ctx, "sdkparcelables").
Colin Cross3047fa22019-04-18 10:56:44 -0700557 Input(jar).
558 Output(aidl)
559
560 aidls = append(aidls, aidl)
561 }
562 }
563
564 combinedAidl := sdkFrameworkAidlPath(ctx)
565 tempPath := combinedAidl.ReplaceExtension(ctx, "aidl.tmp")
566
567 rule.Command().
568 Text("rm -f").Output(tempPath)
569 rule.Command().
570 Text("cat").
571 Inputs(aidls).
572 Text("| sort -u >").
573 Output(tempPath)
574
575 commitChangeForRestat(rule, tempPath, combinedAidl)
576
577 rule.Build(pctx, ctx, "framework_aidl", "generate framework.aidl")
578}
579
580func sdkFrameworkAidlPath(ctx android.PathContext) android.OutputPath {
581 return ctx.Config().Once(sdkFrameworkAidlPathKey, func() interface{} {
582 return android.PathForOutput(ctx, "framework.aidl")
583 }).(android.OutputPath)
584}
585
Colin Cross10932872019-04-18 14:27:12 -0700586// Create api_fingerprint.txt
587func createAPIFingerprint(ctx android.SingletonContext) {
Jiyong Park71b519d2019-04-18 17:25:49 +0900588 out := ApiFingerprintPath(ctx)
Colin Cross10932872019-04-18 14:27:12 -0700589
590 rule := android.NewRuleBuilder()
591
592 rule.Command().
593 Text("rm -f").Output(out)
594 cmd := rule.Command()
595
596 if ctx.Config().PlatformSdkCodename() == "REL" {
597 cmd.Text("echo REL >").Output(out)
598 } else if ctx.Config().IsPdkBuild() {
599 // TODO: get this from the PDK artifacts?
600 cmd.Text("echo PDK >").Output(out)
601 } else if !ctx.Config().UnbundledBuildUsePrebuiltSdks() {
602 in, err := ctx.GlobWithDeps("frameworks/base/api/*current.txt", nil)
603 if err != nil {
604 ctx.Errorf("error globbing API files: %s", err)
605 }
606
607 cmd.Text("cat").
608 Inputs(android.PathsForSource(ctx, in)).
Elliott Hughes34b49d12019-09-06 14:42:24 -0700609 Text("| md5sum | cut -d' ' -f1 >").
Colin Cross10932872019-04-18 14:27:12 -0700610 Output(out)
611 } else {
612 // Unbundled build
613 // TODO: use a prebuilt api_fingerprint.txt from prebuilts/sdk/current.txt once we have one
614 cmd.Text("echo").
615 Flag(ctx.Config().PlatformPreviewSdkVersion()).
616 Text(">").
617 Output(out)
618 }
619
620 rule.Build(pctx, ctx, "api_fingerprint", "generate api_fingerprint.txt")
621}
622
Jiyong Park71b519d2019-04-18 17:25:49 +0900623func ApiFingerprintPath(ctx android.PathContext) android.OutputPath {
Colin Cross10932872019-04-18 14:27:12 -0700624 return ctx.Config().Once(apiFingerprintPathKey, func() interface{} {
625 return android.PathForOutput(ctx, "api_fingerprint.txt")
626 }).(android.OutputPath)
627}
628
629func sdkMakeVars(ctx android.MakeVarsContext) {
630 if ctx.Config().UnbundledBuildUsePrebuiltSdks() || ctx.Config().IsPdkBuild() {
Colin Cross3047fa22019-04-18 10:56:44 -0700631 return
632 }
633
634 ctx.Strict("FRAMEWORK_AIDL", sdkFrameworkAidlPath(ctx).String())
Jiyong Park71b519d2019-04-18 17:25:49 +0900635 ctx.Strict("API_FINGERPRINT", ApiFingerprintPath(ctx).String())
Colin Cross98fd5742019-01-09 23:04:25 -0800636}