Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 1 | // Copyright 2018 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 | |
| 15 | package java |
| 16 | |
| 17 | import ( |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 18 | "fmt" |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 19 | "path" |
Anton Hansson | 370fd0b | 2021-01-22 15:05:04 +0000 | [diff] [blame] | 20 | "strconv" |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 21 | "strings" |
| 22 | |
| 23 | "github.com/google/blueprint/proptools" |
Colin Cross | 17dec17 | 2020-05-14 18:05:32 -0700 | [diff] [blame] | 24 | |
| 25 | "android/soong/android" |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 26 | "android/soong/genrule" |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 27 | ) |
| 28 | |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 29 | func init() { |
Paul Duffin | a48f758 | 2019-12-19 11:25:19 +0000 | [diff] [blame] | 30 | RegisterPrebuiltApisBuildComponents(android.InitRegistrationContext) |
| 31 | } |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 32 | |
Paul Duffin | a48f758 | 2019-12-19 11:25:19 +0000 | [diff] [blame] | 33 | func RegisterPrebuiltApisBuildComponents(ctx android.RegistrationContext) { |
| 34 | ctx.RegisterModuleType("prebuilt_apis", PrebuiltApisFactory) |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 35 | } |
| 36 | |
Sundong Ahn | 27eecb9 | 2018-06-21 13:47:17 +0900 | [diff] [blame] | 37 | type prebuiltApisProperties struct { |
| 38 | // list of api version directories |
| 39 | Api_dirs []string |
Liz Kammer | 2d2fd85 | 2020-08-12 14:42:30 -0700 | [diff] [blame] | 40 | |
Anton Hansson | 3a3f169 | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 41 | // Directory containing finalized api txt files for extension versions. |
| 42 | // Extension versions higher than the base sdk extension version will |
| 43 | // be assumed to be finalized later than all Api_dirs. |
| 44 | Extensions_dir *string |
| 45 | |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 46 | // The next API directory can optionally point to a directory where |
| 47 | // files incompatibility-tracking files are stored for the current |
| 48 | // "in progress" API. Each module present in one of the api_dirs will have |
| 49 | // a <module>-incompatibilities.api.<scope>.latest module created. |
| 50 | Next_api_dir *string |
| 51 | |
Liz Kammer | 2d2fd85 | 2020-08-12 14:42:30 -0700 | [diff] [blame] | 52 | // The sdk_version of java_import modules generated based on jar files. |
| 53 | // Defaults to "current" |
| 54 | Imports_sdk_version *string |
Liz Kammer | 4e7f260 | 2020-09-02 08:37:49 -0700 | [diff] [blame] | 55 | |
| 56 | // If set to true, compile dex for java_import modules. Defaults to false. |
| 57 | Imports_compile_dex *bool |
Sundong Ahn | 27eecb9 | 2018-06-21 13:47:17 +0900 | [diff] [blame] | 58 | } |
| 59 | |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 60 | type prebuiltApis struct { |
| 61 | android.ModuleBase |
Sundong Ahn | 27eecb9 | 2018-06-21 13:47:17 +0900 | [diff] [blame] | 62 | properties prebuiltApisProperties |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 63 | } |
| 64 | |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 65 | func (module *prebuiltApis) GenerateAndroidBuildActions(ctx android.ModuleContext) { |
| 66 | // no need to implement |
| 67 | } |
| 68 | |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 69 | // parsePrebuiltPath parses the relevant variables out of a variety of paths, e.g. |
| 70 | // <version>/<scope>/<module>.jar |
| 71 | // <version>/<scope>/api/<module>.txt |
Todd Lee | 2ec7e1c | 2023-08-25 18:02:13 +0000 | [diff] [blame] | 72 | // *Note when using incremental platform API, <version> may be of the form MM.m where MM is the |
| 73 | // API level and m is an incremental release, otherwise <version> is a single integer corresponding to the API level only. |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 74 | // extensions/<version>/<scope>/<module>.jar |
| 75 | // extensions/<version>/<scope>/api/<module>.txt |
| 76 | func parsePrebuiltPath(ctx android.LoadHookContext, p string) (module string, version string, scope string) { |
| 77 | elements := strings.Split(p, "/") |
Sundong Ahn | a01c2a5 | 2018-06-07 21:42:16 +0900 | [diff] [blame] | 78 | |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 79 | scopeIdx := len(elements) - 2 |
| 80 | if elements[scopeIdx] == "api" { |
| 81 | scopeIdx-- |
| 82 | } |
| 83 | scope = elements[scopeIdx] |
| 84 | if scope != "core" && scope != "public" && scope != "system" && scope != "test" && scope != "module-lib" && scope != "system-server" { |
| 85 | ctx.ModuleErrorf("invalid scope %q found in path: %q", scope, p) |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 86 | return |
| 87 | } |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 88 | version = elements[scopeIdx-1] |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 89 | |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 90 | module = strings.TrimSuffix(path.Base(p), path.Ext(p)) |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 91 | return |
| 92 | } |
| 93 | |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 94 | // parseFinalizedPrebuiltPath is like parsePrebuiltPath, but verifies the version is numeric (a finalized version). |
Mårten Kongstad | 7ceaad1 | 2025-03-03 09:10:01 +0100 | [diff] [blame^] | 95 | func parseFinalizedPrebuiltPath(ctx android.LoadHookContext, p string) (module string, version int, release int, scope string) { |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 96 | module, v, scope := parsePrebuiltPath(ctx, p) |
Mårten Kongstad | 7ceaad1 | 2025-03-03 09:10:01 +0100 | [diff] [blame^] | 97 | |
| 98 | // assume a major.minor version code |
| 99 | parts := strings.Split(v, ".") |
| 100 | if len(parts) == 2 { |
Todd Lee | 2ec7e1c | 2023-08-25 18:02:13 +0000 | [diff] [blame] | 101 | sdk, sdk_err := strconv.Atoi(parts[0]) |
| 102 | qpr, qpr_err := strconv.Atoi(parts[1]) |
| 103 | if sdk_err != nil || qpr_err != nil { |
Mårten Kongstad | 7ceaad1 | 2025-03-03 09:10:01 +0100 | [diff] [blame^] | 104 | ctx.ModuleErrorf("Unable to read major.minor version for prebuilt api '%v'", v) |
Todd Lee | 2ec7e1c | 2023-08-25 18:02:13 +0000 | [diff] [blame] | 105 | return |
| 106 | } |
| 107 | version = sdk |
| 108 | release = qpr |
| 109 | return |
| 110 | } |
Mårten Kongstad | 7ceaad1 | 2025-03-03 09:10:01 +0100 | [diff] [blame^] | 111 | |
| 112 | // assume a legacy integer only api level |
Todd Lee | 2ec7e1c | 2023-08-25 18:02:13 +0000 | [diff] [blame] | 113 | release = 0 |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 114 | version, err := strconv.Atoi(v) |
| 115 | if err != nil { |
Mårten Kongstad | 7ceaad1 | 2025-03-03 09:10:01 +0100 | [diff] [blame^] | 116 | ctx.ModuleErrorf("Unable to read API level for prebuilt api '%v'", v) |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 117 | return |
| 118 | } |
| 119 | return |
Colin Cross | 17dec17 | 2020-05-14 18:05:32 -0700 | [diff] [blame] | 120 | } |
| 121 | |
Jihoon Kang | 28c9657 | 2024-09-11 23:44:44 +0000 | [diff] [blame] | 122 | func prebuiltApiModuleName(moduleName, module, scope, version string) string { |
| 123 | return fmt.Sprintf("%s_%s_%s_%s", moduleName, scope, version, module) |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 124 | } |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 125 | func createImport(mctx android.LoadHookContext, module, scope, version, path, sdkVersion string, compileDex bool) { |
Sundong Ahn | a01c2a5 | 2018-06-07 21:42:16 +0900 | [diff] [blame] | 126 | props := struct { |
Colin Cross | 8ff1058 | 2023-12-07 13:10:56 -0800 | [diff] [blame] | 127 | Name *string |
| 128 | Jars []string |
| 129 | Sdk_version *string |
| 130 | Installable *bool |
| 131 | Compile_dex *bool |
Romain Jobredeaux | 8242b43 | 2023-05-04 10:16:26 -0400 | [diff] [blame] | 132 | }{ |
Jihoon Kang | 28c9657 | 2024-09-11 23:44:44 +0000 | [diff] [blame] | 133 | Name: proptools.StringPtr(prebuiltApiModuleName(mctx.ModuleName(), module, scope, version)), |
Romain Jobredeaux | 8242b43 | 2023-05-04 10:16:26 -0400 | [diff] [blame] | 134 | Jars: []string{path}, |
| 135 | Sdk_version: proptools.StringPtr(sdkVersion), |
| 136 | Installable: proptools.BoolPtr(false), |
| 137 | Compile_dex: proptools.BoolPtr(compileDex), |
| 138 | } |
Colin Cross | 84dfc3d | 2019-09-25 11:33:01 -0700 | [diff] [blame] | 139 | mctx.CreateModule(ImportFactory, &props) |
Sundong Ahn | a01c2a5 | 2018-06-07 21:42:16 +0900 | [diff] [blame] | 140 | } |
| 141 | |
Anton Hansson | c79d412 | 2021-02-17 14:21:33 +0000 | [diff] [blame] | 142 | func createApiModule(mctx android.LoadHookContext, name string, path string) { |
| 143 | genruleProps := struct { |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 144 | Name *string |
| 145 | Srcs []string |
Anton Hansson | c79d412 | 2021-02-17 14:21:33 +0000 | [diff] [blame] | 146 | Out []string |
| 147 | Cmd *string |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 148 | }{} |
Anton Hansson | c79d412 | 2021-02-17 14:21:33 +0000 | [diff] [blame] | 149 | genruleProps.Name = proptools.StringPtr(name) |
| 150 | genruleProps.Srcs = []string{path} |
| 151 | genruleProps.Out = []string{name} |
| 152 | genruleProps.Cmd = proptools.StringPtr("cp $(in) $(out)") |
| 153 | mctx.CreateModule(genrule.GenRuleFactory, &genruleProps) |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 154 | } |
| 155 | |
Jihoon Kang | 748a24d | 2024-03-20 21:29:39 +0000 | [diff] [blame] | 156 | func createCombinedApiFilegroupModule(mctx android.LoadHookContext, name string, srcs []string) { |
| 157 | filegroupProps := struct { |
| 158 | Name *string |
| 159 | Srcs []string |
| 160 | }{} |
| 161 | filegroupProps.Name = proptools.StringPtr(name) |
| 162 | |
| 163 | var transformedSrcs []string |
| 164 | for _, src := range srcs { |
| 165 | transformedSrcs = append(transformedSrcs, ":"+src) |
| 166 | } |
| 167 | filegroupProps.Srcs = transformedSrcs |
| 168 | mctx.CreateModule(android.FileGroupFactory, &filegroupProps) |
| 169 | } |
| 170 | |
Gurpreet Singh | daa314a | 2023-04-21 16:30:03 +0000 | [diff] [blame] | 171 | func createLatestApiModuleExtensionVersionFile(mctx android.LoadHookContext, name string, version string) { |
| 172 | genruleProps := struct { |
| 173 | Name *string |
| 174 | Srcs []string |
| 175 | Out []string |
| 176 | Cmd *string |
| 177 | }{} |
| 178 | genruleProps.Name = proptools.StringPtr(name) |
| 179 | genruleProps.Out = []string{name} |
| 180 | genruleProps.Cmd = proptools.StringPtr("echo " + version + " > $(out)") |
| 181 | mctx.CreateModule(genrule.GenRuleFactory, &genruleProps) |
| 182 | } |
| 183 | |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 184 | func createEmptyFile(mctx android.LoadHookContext, name string) { |
| 185 | props := struct { |
| 186 | Name *string |
| 187 | Cmd *string |
| 188 | Out []string |
| 189 | }{} |
| 190 | props.Name = proptools.StringPtr(name) |
| 191 | props.Out = []string{name} |
| 192 | props.Cmd = proptools.StringPtr("touch $(genDir)/" + name) |
| 193 | mctx.CreateModule(genrule.GenRuleFactory, &props) |
| 194 | } |
| 195 | |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 196 | // globApiDirs collects all the files in all api_dirs and all scopes that match the given glob, e.g. '*.jar' or 'api/*.txt'. |
| 197 | // <api-dir>/<scope>/<glob> for all api-dir and scope. |
| 198 | func globApiDirs(mctx android.LoadHookContext, p *prebuiltApis, api_dir_glob string) []string { |
Sundong Ahn | 27eecb9 | 2018-06-21 13:47:17 +0900 | [diff] [blame] | 199 | var files []string |
Liz Kammer | 2d2fd85 | 2020-08-12 14:42:30 -0700 | [diff] [blame] | 200 | for _, apiver := range p.properties.Api_dirs { |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 201 | files = append(files, globScopeDir(mctx, apiver, api_dir_glob)...) |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 202 | } |
| 203 | return files |
| 204 | } |
| 205 | |
Anton Hansson | 3a3f169 | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 206 | // globExtensionDirs collects all the files under the extension dir (for all versions and scopes) that match the given glob |
| 207 | // <extension-dir>/<version>/<scope>/<glob> for all version and scope. |
| 208 | func globExtensionDirs(mctx android.LoadHookContext, p *prebuiltApis, extension_dir_glob string) []string { |
| 209 | // <extensions-dir>/<num>/<extension-dir-glob> |
| 210 | return globScopeDir(mctx, *p.properties.Extensions_dir+"/*", extension_dir_glob) |
| 211 | } |
| 212 | |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 213 | // globScopeDir collects all the files in the given subdir across all scopes that match the given glob, e.g. '*.jar' or 'api/*.txt'. |
| 214 | // <subdir>/<scope>/<glob> for all scope. |
| 215 | func globScopeDir(mctx android.LoadHookContext, subdir string, subdir_glob string) []string { |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 216 | var files []string |
| 217 | dir := mctx.ModuleDir() + "/" + subdir |
| 218 | for _, scope := range []string{"public", "system", "test", "core", "module-lib", "system-server"} { |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 219 | glob := fmt.Sprintf("%s/%s/%s", dir, scope, subdir_glob) |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 220 | vfiles, err := mctx.GlobWithDeps(glob, nil) |
| 221 | if err != nil { |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 222 | mctx.ModuleErrorf("failed to glob %s files under %q: %s", subdir_glob, dir+"/"+scope, err) |
Sundong Ahn | 27eecb9 | 2018-06-21 13:47:17 +0900 | [diff] [blame] | 223 | } |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 224 | files = append(files, vfiles...) |
Sundong Ahn | a01c2a5 | 2018-06-07 21:42:16 +0900 | [diff] [blame] | 225 | } |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 226 | for i, f := range files { |
| 227 | files[i] = strings.TrimPrefix(f, mctx.ModuleDir()+"/") |
| 228 | } |
Sundong Ahn | 8faab8a | 2019-02-14 11:49:24 +0900 | [diff] [blame] | 229 | return files |
| 230 | } |
| 231 | |
Liz Kammer | 2d2fd85 | 2020-08-12 14:42:30 -0700 | [diff] [blame] | 232 | func prebuiltSdkStubs(mctx android.LoadHookContext, p *prebuiltApis) { |
Sundong Ahn | 8faab8a | 2019-02-14 11:49:24 +0900 | [diff] [blame] | 233 | // <apiver>/<scope>/<module>.jar |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 234 | files := globApiDirs(mctx, p, "*.jar") |
Liz Kammer | 2d2fd85 | 2020-08-12 14:42:30 -0700 | [diff] [blame] | 235 | |
Liz Kammer | 4e7f260 | 2020-09-02 08:37:49 -0700 | [diff] [blame] | 236 | sdkVersion := proptools.StringDefault(p.properties.Imports_sdk_version, "current") |
| 237 | compileDex := proptools.BoolDefault(p.properties.Imports_compile_dex, false) |
Sundong Ahn | a01c2a5 | 2018-06-07 21:42:16 +0900 | [diff] [blame] | 238 | |
| 239 | for _, f := range files { |
| 240 | // create a Import module for each jar file |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 241 | module, version, scope := parsePrebuiltPath(mctx, f) |
| 242 | createImport(mctx, module, scope, version, f, sdkVersion, compileDex) |
Paul Duffin | b077bcc | 2021-10-28 13:27:37 +0100 | [diff] [blame] | 243 | |
| 244 | if module == "core-for-system-modules" { |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 245 | createSystemModules(mctx, version, scope) |
Paul Duffin | b077bcc | 2021-10-28 13:27:37 +0100 | [diff] [blame] | 246 | } |
Sundong Ahn | a01c2a5 | 2018-06-07 21:42:16 +0900 | [diff] [blame] | 247 | } |
| 248 | } |
| 249 | |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 250 | func createSystemModules(mctx android.LoadHookContext, version, scope string) { |
Colin Cross | 17dec17 | 2020-05-14 18:05:32 -0700 | [diff] [blame] | 251 | props := struct { |
| 252 | Name *string |
| 253 | Libs []string |
| 254 | }{} |
Jihoon Kang | 28c9657 | 2024-09-11 23:44:44 +0000 | [diff] [blame] | 255 | props.Name = proptools.StringPtr(prebuiltApiModuleName(mctx.ModuleName(), "system_modules", scope, version)) |
| 256 | props.Libs = append(props.Libs, prebuiltApiModuleName(mctx.ModuleName(), "core-for-system-modules", scope, version)) |
Colin Cross | 17dec17 | 2020-05-14 18:05:32 -0700 | [diff] [blame] | 257 | |
Paul Duffin | d6c2a65 | 2021-03-11 07:56:22 +0000 | [diff] [blame] | 258 | mctx.CreateModule(systemModulesImportFactory, &props) |
Colin Cross | 17dec17 | 2020-05-14 18:05:32 -0700 | [diff] [blame] | 259 | } |
| 260 | |
Paul Duffin | 958806b | 2022-05-16 13:10:47 +0000 | [diff] [blame] | 261 | func PrebuiltApiModuleName(module, scope, version string) string { |
| 262 | return module + ".api." + scope + "." + version |
| 263 | } |
| 264 | |
Jihoon Kang | 748a24d | 2024-03-20 21:29:39 +0000 | [diff] [blame] | 265 | func PrebuiltApiCombinedModuleName(module, scope, version string) string { |
| 266 | return module + ".api.combined." + scope + "." + version |
| 267 | } |
| 268 | |
Liz Kammer | 2d2fd85 | 2020-08-12 14:42:30 -0700 | [diff] [blame] | 269 | func prebuiltApiFiles(mctx android.LoadHookContext, p *prebuiltApis) { |
Sundong Ahn | a01c2a5 | 2018-06-07 21:42:16 +0900 | [diff] [blame] | 270 | // <apiver>/<scope>/api/<module>.txt |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 271 | apiLevelFiles := globApiDirs(mctx, p, "api/*.txt") |
| 272 | if len(apiLevelFiles) == 0 { |
| 273 | mctx.ModuleErrorf("no api file found under %q", mctx.ModuleDir()) |
Sundong Ahn | a01c2a5 | 2018-06-07 21:42:16 +0900 | [diff] [blame] | 274 | } |
Sundong Ahn | a01c2a5 | 2018-06-07 21:42:16 +0900 | [diff] [blame] | 275 | |
Anton Hansson | c79d412 | 2021-02-17 14:21:33 +0000 | [diff] [blame] | 276 | // Create modules for all (<module>, <scope, <version>) triplets, |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 277 | for _, f := range apiLevelFiles { |
Mårten Kongstad | 7ceaad1 | 2025-03-03 09:10:01 +0100 | [diff] [blame^] | 278 | module, version, release, scope := parseFinalizedPrebuiltPath(mctx, f) |
| 279 | if release != 0 { |
| 280 | majorDotMinorVersion := strconv.Itoa(version) + "." + strconv.Itoa(release) |
| 281 | createApiModule(mctx, PrebuiltApiModuleName(module, scope, majorDotMinorVersion), f) |
Todd Lee | 2ec7e1c | 2023-08-25 18:02:13 +0000 | [diff] [blame] | 282 | } else { |
| 283 | createApiModule(mctx, PrebuiltApiModuleName(module, scope, strconv.Itoa(version)), f) |
| 284 | } |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 285 | } |
Sundong Ahn | a01c2a5 | 2018-06-07 21:42:16 +0900 | [diff] [blame] | 286 | |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 287 | // Figure out the latest version of each module/scope |
| 288 | type latestApiInfo struct { |
| 289 | module, scope, path string |
Todd Lee | 2ec7e1c | 2023-08-25 18:02:13 +0000 | [diff] [blame] | 290 | version, release int |
Gurpreet Singh | daa314a | 2023-04-21 16:30:03 +0000 | [diff] [blame] | 291 | isExtensionApiFile bool |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 292 | } |
| 293 | |
Gurpreet Singh | daa314a | 2023-04-21 16:30:03 +0000 | [diff] [blame] | 294 | getLatest := func(files []string, isExtensionApiFile bool) map[string]latestApiInfo { |
Anton Hansson | 3a3f169 | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 295 | m := make(map[string]latestApiInfo) |
| 296 | for _, f := range files { |
Mårten Kongstad | 7ceaad1 | 2025-03-03 09:10:01 +0100 | [diff] [blame^] | 297 | module, version, release, scope := parseFinalizedPrebuiltPath(mctx, f) |
Anton Hansson | 3a3f169 | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 298 | if strings.HasSuffix(module, "incompatibilities") { |
| 299 | continue |
| 300 | } |
| 301 | key := module + "." + scope |
| 302 | info, exists := m[key] |
Todd Lee | 2ec7e1c | 2023-08-25 18:02:13 +0000 | [diff] [blame] | 303 | if !exists || version > info.version || (version == info.version && release > info.release) { |
| 304 | m[key] = latestApiInfo{module, scope, f, version, release, isExtensionApiFile} |
Anton Hansson | 3a3f169 | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 305 | } |
Anton Hansson | 370fd0b | 2021-01-22 15:05:04 +0000 | [diff] [blame] | 306 | } |
Anton Hansson | 3a3f169 | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 307 | return m |
| 308 | } |
Anton Hansson | 370fd0b | 2021-01-22 15:05:04 +0000 | [diff] [blame] | 309 | |
Gurpreet Singh | daa314a | 2023-04-21 16:30:03 +0000 | [diff] [blame] | 310 | latest := getLatest(apiLevelFiles, false) |
Anton Hansson | 3a3f169 | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 311 | if p.properties.Extensions_dir != nil { |
| 312 | extensionApiFiles := globExtensionDirs(mctx, p, "api/*.txt") |
Gurpreet Singh | daa314a | 2023-04-21 16:30:03 +0000 | [diff] [blame] | 313 | for k, v := range getLatest(extensionApiFiles, true) { |
Anton Hansson | 7212dbe | 2022-09-20 17:08:49 +0000 | [diff] [blame] | 314 | if _, exists := latest[k]; !exists { |
| 315 | mctx.ModuleErrorf("Module %v finalized for extension %d but never during an API level; likely error", v.module, v.version) |
Anton Hansson | 3a3f169 | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 316 | } |
Anton Hansson | 7212dbe | 2022-09-20 17:08:49 +0000 | [diff] [blame] | 317 | // The extension version is always at least as new as the last sdk int version (potentially identical) |
| 318 | latest[k] = v |
Sundong Ahn | a01c2a5 | 2018-06-07 21:42:16 +0900 | [diff] [blame] | 319 | } |
| 320 | } |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 321 | |
Anton Hansson | 370fd0b | 2021-01-22 15:05:04 +0000 | [diff] [blame] | 322 | // Sort the keys in order to make build.ninja stable |
Jihoon Kang | 748a24d | 2024-03-20 21:29:39 +0000 | [diff] [blame] | 323 | sortedLatestKeys := android.SortedKeys(latest) |
| 324 | |
| 325 | for _, k := range sortedLatestKeys { |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 326 | info := latest[k] |
Paul Duffin | 958806b | 2022-05-16 13:10:47 +0000 | [diff] [blame] | 327 | name := PrebuiltApiModuleName(info.module, info.scope, "latest") |
Gurpreet Singh | daa314a | 2023-04-21 16:30:03 +0000 | [diff] [blame] | 328 | latestExtensionVersionModuleName := PrebuiltApiModuleName(info.module, info.scope, "latest.extension_version") |
| 329 | if info.isExtensionApiFile { |
| 330 | createLatestApiModuleExtensionVersionFile(mctx, latestExtensionVersionModuleName, strconv.Itoa(info.version)) |
| 331 | } else { |
| 332 | createLatestApiModuleExtensionVersionFile(mctx, latestExtensionVersionModuleName, "-1") |
| 333 | } |
Anton Hansson | c79d412 | 2021-02-17 14:21:33 +0000 | [diff] [blame] | 334 | createApiModule(mctx, name, info.path) |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 335 | } |
| 336 | |
| 337 | // Create incompatibilities tracking files for all modules, if we have a "next" api. |
Jaewoong Jung | 1a97ee0 | 2021-03-09 13:25:02 -0800 | [diff] [blame] | 338 | incompatibilities := make(map[string]bool) |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 339 | if nextApiDir := String(p.properties.Next_api_dir); nextApiDir != "" { |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 340 | files := globScopeDir(mctx, nextApiDir, "api/*incompatibilities.txt") |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 341 | for _, f := range files { |
Anton Hansson | 377318b | 2022-02-15 12:55:11 +0000 | [diff] [blame] | 342 | filename, _, scope := parsePrebuiltPath(mctx, f) |
Anton Hansson | fa5e6b5 | 2021-04-13 19:09:48 +0100 | [diff] [blame] | 343 | referencedModule := strings.TrimSuffix(filename, "-incompatibilities") |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 344 | |
Paul Duffin | 958806b | 2022-05-16 13:10:47 +0000 | [diff] [blame] | 345 | createApiModule(mctx, PrebuiltApiModuleName(referencedModule+"-incompatibilities", scope, "latest"), f) |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 346 | |
| 347 | incompatibilities[referencedModule+"."+scope] = true |
| 348 | } |
Jaewoong Jung | 1a97ee0 | 2021-03-09 13:25:02 -0800 | [diff] [blame] | 349 | } |
| 350 | // Create empty incompatibilities files for remaining modules |
Jihoon Kang | 748a24d | 2024-03-20 21:29:39 +0000 | [diff] [blame] | 351 | // If the incompatibility module has been created, create a corresponding combined module |
| 352 | for _, k := range sortedLatestKeys { |
Jaewoong Jung | 1a97ee0 | 2021-03-09 13:25:02 -0800 | [diff] [blame] | 353 | if _, ok := incompatibilities[k]; !ok { |
Paul Duffin | 958806b | 2022-05-16 13:10:47 +0000 | [diff] [blame] | 354 | createEmptyFile(mctx, PrebuiltApiModuleName(latest[k].module+"-incompatibilities", latest[k].scope, "latest")) |
Anton Hansson | 20ce41d | 2021-01-22 15:05:32 +0000 | [diff] [blame] | 355 | } |
Sundong Ahn | a01c2a5 | 2018-06-07 21:42:16 +0900 | [diff] [blame] | 356 | } |
Jihoon Kang | 748a24d | 2024-03-20 21:29:39 +0000 | [diff] [blame] | 357 | |
| 358 | // Create combined latest api and removed api files modules. |
| 359 | // The combined modules list all api files of the api scope and its subset api scopes. |
| 360 | for _, k := range sortedLatestKeys { |
| 361 | info := latest[k] |
| 362 | name := PrebuiltApiCombinedModuleName(info.module, info.scope, "latest") |
| 363 | |
Jihoon Kang | 5623e54 | 2024-01-31 23:27:26 +0000 | [diff] [blame] | 364 | // Iterate until the currentApiScope does not extend any other api scopes |
| 365 | // i.e. is not a superset of any other api scopes |
| 366 | // the relationship between the api scopes is defined in java/sdk_library.go |
Jihoon Kang | 748a24d | 2024-03-20 21:29:39 +0000 | [diff] [blame] | 367 | var srcs []string |
| 368 | currentApiScope := scopeByName[info.scope] |
Jihoon Kang | 5623e54 | 2024-01-31 23:27:26 +0000 | [diff] [blame] | 369 | for currentApiScope != nil { |
| 370 | if _, ok := latest[fmt.Sprintf("%s.%s", info.module, currentApiScope.name)]; ok { |
| 371 | srcs = append(srcs, PrebuiltApiModuleName(info.module, currentApiScope.name, "latest")) |
| 372 | } |
| 373 | currentApiScope = currentApiScope.extends |
| 374 | } |
Jihoon Kang | 748a24d | 2024-03-20 21:29:39 +0000 | [diff] [blame] | 375 | |
Jihoon Kang | 5623e54 | 2024-01-31 23:27:26 +0000 | [diff] [blame] | 376 | // srcs is currently listed in the order from the widest api scope to the narrowest api scopes |
| 377 | // e.g. module lib -> system -> public |
| 378 | // In order to pass the files in metalava from the narrowest api scope to the widest api scope, |
| 379 | // the list has to be reversed. |
| 380 | android.ReverseSliceInPlace(srcs) |
Jihoon Kang | 748a24d | 2024-03-20 21:29:39 +0000 | [diff] [blame] | 381 | createCombinedApiFilegroupModule(mctx, name, srcs) |
| 382 | } |
Sundong Ahn | a01c2a5 | 2018-06-07 21:42:16 +0900 | [diff] [blame] | 383 | } |
| 384 | |
Paul Duffin | d4c0356 | 2020-04-09 17:15:44 +0100 | [diff] [blame] | 385 | func createPrebuiltApiModules(mctx android.LoadHookContext) { |
Liz Kammer | 2d2fd85 | 2020-08-12 14:42:30 -0700 | [diff] [blame] | 386 | if p, ok := mctx.Module().(*prebuiltApis); ok { |
| 387 | prebuiltApiFiles(mctx, p) |
| 388 | prebuiltSdkStubs(mctx, p) |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 389 | } |
| 390 | } |
| 391 | |
Anton Hansson | c79d412 | 2021-02-17 14:21:33 +0000 | [diff] [blame] | 392 | // prebuilt_apis is a meta-module that generates modules for all API txt files |
| 393 | // found under the directory where the Android.bp is located. |
Jaewoong Jung | 5fb5b2a | 2019-03-21 10:48:25 -0700 | [diff] [blame] | 394 | // Specifically, an API file located at ./<ver>/<scope>/api/<module>.txt |
Anton Hansson | c79d412 | 2021-02-17 14:21:33 +0000 | [diff] [blame] | 395 | // generates a module named <module>-api.<scope>.<ver>. |
Jaewoong Jung | 5fb5b2a | 2019-03-21 10:48:25 -0700 | [diff] [blame] | 396 | // |
| 397 | // It also creates <module>-api.<scope>.latest for the latest <ver>. |
Paul Duffin | d4c0356 | 2020-04-09 17:15:44 +0100 | [diff] [blame] | 398 | // |
| 399 | // Similarly, it generates a java_import for all API .jar files found under the |
| 400 | // directory where the Android.bp is located. Specifically, an API file located |
| 401 | // at ./<ver>/<scope>/api/<module>.jar generates a java_import module named |
Colin Cross | 17dec17 | 2020-05-14 18:05:32 -0700 | [diff] [blame] | 402 | // <prebuilt-api-module>_<scope>_<ver>_<module>, and for SDK versions >= 30 |
| 403 | // a java_system_modules module named |
| 404 | // <prebuilt-api-module>_public_<ver>_system_modules |
Inseob Kim | c0907f1 | 2019-02-08 21:00:45 +0900 | [diff] [blame] | 405 | func PrebuiltApisFactory() android.Module { |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 406 | module := &prebuiltApis{} |
Sundong Ahn | 27eecb9 | 2018-06-21 13:47:17 +0900 | [diff] [blame] | 407 | module.AddProperties(&module.properties) |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 408 | android.InitAndroidModule(module) |
Paul Duffin | d4c0356 | 2020-04-09 17:15:44 +0100 | [diff] [blame] | 409 | android.AddLoadHook(module, createPrebuiltApiModules) |
Jiyong Park | 58c518b | 2018-05-12 22:29:12 +0900 | [diff] [blame] | 410 | return module |
| 411 | } |