blob: 9dfb38bf48a60a9629d4c62ecb1bead02a0e76f9 [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 (
18 "android/soong/android"
19 "android/soong/java/config"
20 "fmt"
21 "path/filepath"
Colin Cross10932872019-04-18 14:27:12 -070022 "runtime"
Colin Cross98fd5742019-01-09 23:04:25 -080023 "sort"
Colin Crossfb6d7812019-01-09 22:17:55 -080024 "strconv"
25 "strings"
Colin Cross3047fa22019-04-18 10:56:44 -070026
27 "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 {
Paul Duffin250e6192019-06-07 10:44:37 +010041 // sdkVersion returns the sdk_version property of the current module, or an empty string if it is not set.
Colin Crossfb6d7812019-01-09 22:17:55 -080042 sdkVersion() string
43 // minSdkVersion returns the min_sdk_version property of the current module, or sdkVersion() if it is not set.
44 minSdkVersion() string
45 // targetSdkVersion returns the target_sdk_version property of the current module, or sdkVersion() if it is not set.
46 targetSdkVersion() string
Paul Duffin250e6192019-06-07 10:44:37 +010047
48 // Temporarily provide access to the no_standard_libs property (where present).
49 noStandardLibs() bool
50
51 // Temporarily provide access to the no_frameworks_libs property (where present).
52 noFrameworkLibs() bool
Colin Crossfb6d7812019-01-09 22:17:55 -080053}
54
Colin Cross0ea8ba82019-06-06 14:33:29 -070055func sdkVersionOrDefault(ctx android.BaseModuleContext, v string) string {
Colin Crossfb6d7812019-01-09 22:17:55 -080056 switch v {
Paul Duffin52d398a2019-06-11 12:31:14 +010057 case "", "none", "current", "system_current", "test_current", "core_current":
Colin Crossfb6d7812019-01-09 22:17:55 -080058 return ctx.Config().DefaultAppTargetSdk()
59 default:
60 return v
61 }
62}
63
64// Returns a sdk version as a number. For modules targeting an unreleased SDK (meaning it does not yet have a number)
65// it returns android.FutureApiLevel (10000).
Colin Cross0ea8ba82019-06-06 14:33:29 -070066func sdkVersionToNumber(ctx android.BaseModuleContext, v string) (int, error) {
Colin Crossfb6d7812019-01-09 22:17:55 -080067 switch v {
Paul Duffin52d398a2019-06-11 12:31:14 +010068 case "", "none", "current", "test_current", "system_current", "core_current":
Colin Crossfb6d7812019-01-09 22:17:55 -080069 return ctx.Config().DefaultAppTargetSdkInt(), nil
70 default:
71 n := android.GetNumericSdkVersion(v)
72 if i, err := strconv.Atoi(n); err != nil {
73 return -1, fmt.Errorf("invalid sdk version %q", n)
74 } else {
75 return i, nil
76 }
77 }
78}
79
Colin Cross0ea8ba82019-06-06 14:33:29 -070080func sdkVersionToNumberAsString(ctx android.BaseModuleContext, v string) (string, error) {
Colin Crossfb6d7812019-01-09 22:17:55 -080081 n, err := sdkVersionToNumber(ctx, v)
82 if err != nil {
83 return "", err
84 }
85 return strconv.Itoa(n), nil
86}
87
Colin Cross0ea8ba82019-06-06 14:33:29 -070088func decodeSdkDep(ctx android.BaseModuleContext, sdkContext sdkContext) sdkDep {
Colin Crossfb6d7812019-01-09 22:17:55 -080089 v := sdkContext.sdkVersion()
Colin Cross98fd5742019-01-09 23:04:25 -080090 // For PDK builds, use the latest SDK version instead of "current"
91 if ctx.Config().IsPdkBuild() && (v == "" || v == "current") {
Colin Cross3047fa22019-04-18 10:56:44 -070092 sdkVersions := ctx.Config().Get(sdkVersionsKey).([]int)
Colin Cross98fd5742019-01-09 23:04:25 -080093 latestSdkVersion := 0
94 if len(sdkVersions) > 0 {
95 latestSdkVersion = sdkVersions[len(sdkVersions)-1]
96 }
97 v = strconv.Itoa(latestSdkVersion)
98 }
99
Colin Crossff0daf42019-04-02 16:10:56 -0700100 numericSdkVersion, err := sdkVersionToNumber(ctx, v)
Colin Crossfb6d7812019-01-09 22:17:55 -0800101 if err != nil {
102 ctx.PropertyErrorf("sdk_version", "%s", err)
103 return sdkDep{}
104 }
105
Colin Crossfb6d7812019-01-09 22:17:55 -0800106 toPrebuilt := func(sdk string) sdkDep {
107 var api, v string
108 if strings.Contains(sdk, "_") {
109 t := strings.Split(sdk, "_")
110 api = t[0]
111 v = t[1]
112 } else {
113 api = "public"
114 v = sdk
115 }
116 dir := filepath.Join("prebuilts", "sdk", v, api)
117 jar := filepath.Join(dir, "android.jar")
118 // There's no aidl for other SDKs yet.
119 // TODO(77525052): Add aidl files for other SDKs too.
120 public_dir := filepath.Join("prebuilts", "sdk", v, "public")
121 aidl := filepath.Join(public_dir, "framework.aidl")
122 jarPath := android.ExistentPathForSource(ctx, jar)
123 aidlPath := android.ExistentPathForSource(ctx, aidl)
124 lambdaStubsPath := android.PathForSource(ctx, config.SdkLambdaStubsPath)
125
126 if (!jarPath.Valid() || !aidlPath.Valid()) && ctx.Config().AllowMissingDependencies() {
127 return sdkDep{
128 invalidVersion: true,
129 modules: []string{fmt.Sprintf("sdk_%s_%s_android", api, v)},
130 }
131 }
132
133 if !jarPath.Valid() {
134 ctx.PropertyErrorf("sdk_version", "invalid sdk version %q, %q does not exist", v, jar)
135 return sdkDep{}
136 }
137
138 if !aidlPath.Valid() {
139 ctx.PropertyErrorf("sdk_version", "invalid sdk version %q, %q does not exist", v, aidl)
140 return sdkDep{}
141 }
142
143 return sdkDep{
144 useFiles: true,
145 jars: android.Paths{jarPath.Path(), lambdaStubsPath},
Colin Cross3047fa22019-04-18 10:56:44 -0700146 aidl: android.OptionalPathForPath(aidlPath.Path()),
Paul Duffin250e6192019-06-07 10:44:37 +0100147
148 // Pass values straight through for now to match previous behavior.
149 noStandardLibs: sdkContext.noStandardLibs(),
150 noFrameworksLibs: sdkContext.noFrameworkLibs(),
Colin Crossfb6d7812019-01-09 22:17:55 -0800151 }
152 }
153
Colin Cross3047fa22019-04-18 10:56:44 -0700154 toModule := func(m, r string, aidl android.Path) sdkDep {
Colin Crossfb6d7812019-01-09 22:17:55 -0800155 ret := sdkDep{
156 useModule: true,
157 modules: []string{m, config.DefaultLambdaStubsLibrary},
158 systemModules: m + "_system_modules",
159 frameworkResModule: r,
Colin Cross3047fa22019-04-18 10:56:44 -0700160 aidl: android.OptionalPathForPath(aidl),
Paul Duffin250e6192019-06-07 10:44:37 +0100161
162 // Pass values straight through for now to match previous behavior.
163 noStandardLibs: sdkContext.noStandardLibs(),
164 noFrameworksLibs: sdkContext.noFrameworkLibs(),
Colin Crossfb6d7812019-01-09 22:17:55 -0800165 }
Colin Cross3047fa22019-04-18 10:56:44 -0700166
Colin Crossfb6d7812019-01-09 22:17:55 -0800167 if m == "core.current.stubs" {
Neil Fullerba88c412018-10-21 22:57:26 +0100168 ret.systemModules = "core-current-stubs-system-modules"
Colin Crossfb6d7812019-01-09 22:17:55 -0800169 } else if m == "core.platform.api.stubs" {
170 ret.systemModules = "core-platform-api-stubs-system-modules"
171 }
172 return ret
173 }
174
Colin Cross98fd5742019-01-09 23:04:25 -0800175 // Ensures that the specificed system SDK version is one of BOARD_SYSTEMSDK_VERSIONS (for vendor apks)
176 // or PRODUCT_SYSTEMSDK_VERSIONS (for other apks or when BOARD_SYSTEMSDK_VERSIONS is not set)
Colin Crossff0daf42019-04-02 16:10:56 -0700177 if strings.HasPrefix(v, "system_") && numericSdkVersion != android.FutureApiLevel {
Colin Cross98fd5742019-01-09 23:04:25 -0800178 allowed_versions := ctx.DeviceConfig().PlatformSystemSdkVersions()
179 if ctx.DeviceSpecific() || ctx.SocSpecific() {
180 if len(ctx.DeviceConfig().SystemSdkVersions()) > 0 {
181 allowed_versions = ctx.DeviceConfig().SystemSdkVersions()
182 }
183 }
Colin Crossff0daf42019-04-02 16:10:56 -0700184 if len(allowed_versions) > 0 && !android.InList(strconv.Itoa(numericSdkVersion), allowed_versions) {
Colin Cross98fd5742019-01-09 23:04:25 -0800185 ctx.PropertyErrorf("sdk_version", "incompatible sdk version %q. System SDK version should be one of %q",
186 v, allowed_versions)
187 }
188 }
189
Paul Duffin52d398a2019-06-11 12:31:14 +0100190 if ctx.Config().UnbundledBuildUsePrebuiltSdks() && v != "" && v != "none" {
Colin Crossfb6d7812019-01-09 22:17:55 -0800191 return toPrebuilt(v)
192 }
193
194 switch v {
195 case "":
196 return sdkDep{
197 useDefaultLibs: true,
198 frameworkResModule: "framework-res",
Paul Duffin250e6192019-06-07 10:44:37 +0100199
200 // Pass values straight through for now to match previous behavior.
201 noStandardLibs: sdkContext.noStandardLibs(),
202 noFrameworksLibs: sdkContext.noFrameworkLibs(),
Colin Crossfb6d7812019-01-09 22:17:55 -0800203 }
Paul Duffin52d398a2019-06-11 12:31:14 +0100204 case "none":
205 return sdkDep{
206 noStandardLibs: true,
207 }
Colin Crossfb6d7812019-01-09 22:17:55 -0800208 case "current":
Colin Cross3047fa22019-04-18 10:56:44 -0700209 return toModule("android_stubs_current", "framework-res", sdkFrameworkAidlPath(ctx))
Colin Crossfb6d7812019-01-09 22:17:55 -0800210 case "system_current":
Colin Cross3047fa22019-04-18 10:56:44 -0700211 return toModule("android_system_stubs_current", "framework-res", sdkFrameworkAidlPath(ctx))
Colin Crossfb6d7812019-01-09 22:17:55 -0800212 case "test_current":
Colin Cross3047fa22019-04-18 10:56:44 -0700213 return toModule("android_test_stubs_current", "framework-res", sdkFrameworkAidlPath(ctx))
Colin Crossfb6d7812019-01-09 22:17:55 -0800214 case "core_current":
Colin Cross3047fa22019-04-18 10:56:44 -0700215 return toModule("core.current.stubs", "", nil)
Colin Crossfb6d7812019-01-09 22:17:55 -0800216 default:
217 return toPrebuilt(v)
218 }
219}
Colin Cross98fd5742019-01-09 23:04:25 -0800220
Colin Cross3047fa22019-04-18 10:56:44 -0700221func sdkPreSingletonFactory() android.Singleton {
222 return sdkPreSingleton{}
Colin Cross98fd5742019-01-09 23:04:25 -0800223}
224
Colin Cross3047fa22019-04-18 10:56:44 -0700225type sdkPreSingleton struct{}
Colin Cross98fd5742019-01-09 23:04:25 -0800226
Colin Cross3047fa22019-04-18 10:56:44 -0700227func (sdkPreSingleton) GenerateBuildActions(ctx android.SingletonContext) {
Colin Cross98fd5742019-01-09 23:04:25 -0800228 sdkJars, err := ctx.GlobWithDeps("prebuilts/sdk/*/public/android.jar", nil)
229 if err != nil {
230 ctx.Errorf("failed to glob prebuilts/sdk/*/public/android.jar: %s", err.Error())
231 }
232
233 var sdkVersions []int
234 for _, sdkJar := range sdkJars {
235 dir := filepath.Base(filepath.Dir(filepath.Dir(sdkJar)))
236 v, err := strconv.Atoi(dir)
237 if scerr, ok := err.(*strconv.NumError); ok && scerr.Err == strconv.ErrSyntax {
238 continue
239 } else if err != nil {
240 ctx.Errorf("invalid sdk jar %q, %s, %v", sdkJar, err.Error())
241 }
242 sdkVersions = append(sdkVersions, v)
243 }
244
245 sort.Ints(sdkVersions)
246
Colin Cross3047fa22019-04-18 10:56:44 -0700247 ctx.Config().Once(sdkVersionsKey, func() interface{} { return sdkVersions })
248}
249
250func sdkSingletonFactory() android.Singleton {
251 return sdkSingleton{}
252}
253
254type sdkSingleton struct{}
255
256func (sdkSingleton) GenerateBuildActions(ctx android.SingletonContext) {
Colin Cross10932872019-04-18 14:27:12 -0700257 if ctx.Config().UnbundledBuildUsePrebuiltSdks() || ctx.Config().IsPdkBuild() {
Colin Cross3047fa22019-04-18 10:56:44 -0700258 return
259 }
260
Colin Cross10932872019-04-18 14:27:12 -0700261 createSdkFrameworkAidl(ctx)
262 createAPIFingerprint(ctx)
263}
Colin Cross3047fa22019-04-18 10:56:44 -0700264
Colin Cross10932872019-04-18 14:27:12 -0700265// Create framework.aidl by extracting anything that implements android.os.Parcelable from the SDK stubs modules.
266func createSdkFrameworkAidl(ctx android.SingletonContext) {
Colin Cross3047fa22019-04-18 10:56:44 -0700267 stubsModules := []string{
268 "android_stubs_current",
269 "android_test_stubs_current",
270 "android_system_stubs_current",
271 }
272
273 stubsJars := make([]android.Paths, len(stubsModules))
274
275 ctx.VisitAllModules(func(module android.Module) {
276 // Collect dex jar paths for the modules listed above.
277 if j, ok := module.(Dependency); ok {
278 name := ctx.ModuleName(module)
279 if i := android.IndexList(name, stubsModules); i != -1 {
280 stubsJars[i] = j.HeaderJars()
281 }
282 }
283 })
284
285 var missingDeps []string
286
287 for i := range stubsJars {
288 if stubsJars[i] == nil {
289 if ctx.Config().AllowMissingDependencies() {
290 missingDeps = append(missingDeps, stubsModules[i])
291 } else {
292 ctx.Errorf("failed to find dex jar path for module %q",
293 stubsModules[i])
294 }
295 }
296 }
297
298 rule := android.NewRuleBuilder()
299 rule.MissingDeps(missingDeps)
300
301 var aidls android.Paths
302 for _, jars := range stubsJars {
303 for _, jar := range jars {
304 aidl := android.PathForOutput(ctx, "aidl", pathtools.ReplaceExtension(jar.Base(), "aidl"))
305
306 rule.Command().
307 Text("rm -f").Output(aidl)
308 rule.Command().
309 Tool(ctx.Config().HostToolPath(ctx, "sdkparcelables")).
310 Input(jar).
311 Output(aidl)
312
313 aidls = append(aidls, aidl)
314 }
315 }
316
317 combinedAidl := sdkFrameworkAidlPath(ctx)
318 tempPath := combinedAidl.ReplaceExtension(ctx, "aidl.tmp")
319
320 rule.Command().
321 Text("rm -f").Output(tempPath)
322 rule.Command().
323 Text("cat").
324 Inputs(aidls).
325 Text("| sort -u >").
326 Output(tempPath)
327
328 commitChangeForRestat(rule, tempPath, combinedAidl)
329
330 rule.Build(pctx, ctx, "framework_aidl", "generate framework.aidl")
331}
332
333func sdkFrameworkAidlPath(ctx android.PathContext) android.OutputPath {
334 return ctx.Config().Once(sdkFrameworkAidlPathKey, func() interface{} {
335 return android.PathForOutput(ctx, "framework.aidl")
336 }).(android.OutputPath)
337}
338
Colin Cross10932872019-04-18 14:27:12 -0700339// Create api_fingerprint.txt
340func createAPIFingerprint(ctx android.SingletonContext) {
Jiyong Park71b519d2019-04-18 17:25:49 +0900341 out := ApiFingerprintPath(ctx)
Colin Cross10932872019-04-18 14:27:12 -0700342
343 rule := android.NewRuleBuilder()
344
345 rule.Command().
346 Text("rm -f").Output(out)
347 cmd := rule.Command()
348
349 if ctx.Config().PlatformSdkCodename() == "REL" {
350 cmd.Text("echo REL >").Output(out)
351 } else if ctx.Config().IsPdkBuild() {
352 // TODO: get this from the PDK artifacts?
353 cmd.Text("echo PDK >").Output(out)
354 } else if !ctx.Config().UnbundledBuildUsePrebuiltSdks() {
355 in, err := ctx.GlobWithDeps("frameworks/base/api/*current.txt", nil)
356 if err != nil {
357 ctx.Errorf("error globbing API files: %s", err)
358 }
359
360 cmd.Text("cat").
361 Inputs(android.PathsForSource(ctx, in)).
362 Text("|")
363
364 if runtime.GOOS == "darwin" {
365 cmd.Text("md5")
366 } else {
367 cmd.Text("md5sum")
368 }
369
370 cmd.Text("| cut -d' ' -f1 >").
371 Output(out)
372 } else {
373 // Unbundled build
374 // TODO: use a prebuilt api_fingerprint.txt from prebuilts/sdk/current.txt once we have one
375 cmd.Text("echo").
376 Flag(ctx.Config().PlatformPreviewSdkVersion()).
377 Text(">").
378 Output(out)
379 }
380
381 rule.Build(pctx, ctx, "api_fingerprint", "generate api_fingerprint.txt")
382}
383
Jiyong Park71b519d2019-04-18 17:25:49 +0900384func ApiFingerprintPath(ctx android.PathContext) android.OutputPath {
Colin Cross10932872019-04-18 14:27:12 -0700385 return ctx.Config().Once(apiFingerprintPathKey, func() interface{} {
386 return android.PathForOutput(ctx, "api_fingerprint.txt")
387 }).(android.OutputPath)
388}
389
390func sdkMakeVars(ctx android.MakeVarsContext) {
391 if ctx.Config().UnbundledBuildUsePrebuiltSdks() || ctx.Config().IsPdkBuild() {
Colin Cross3047fa22019-04-18 10:56:44 -0700392 return
393 }
394
395 ctx.Strict("FRAMEWORK_AIDL", sdkFrameworkAidlPath(ctx).String())
Jiyong Park71b519d2019-04-18 17:25:49 +0900396 ctx.Strict("API_FINGERPRINT", ApiFingerprintPath(ctx).String())
Colin Cross98fd5742019-01-09 23:04:25 -0800397}