Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 1 | // Copyright 2016 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 cc |
| 16 | |
| 17 | // The platform needs to provide the following artifacts for the NDK: |
| 18 | // 1. Bionic headers. |
| 19 | // 2. Platform API headers. |
| 20 | // 3. NDK stub shared libraries. |
| 21 | // 4. Bionic static libraries. |
| 22 | // |
| 23 | // TODO(danalbert): All of the above need to include NOTICE files. |
| 24 | // |
| 25 | // Components 1 and 2: Headers |
| 26 | // The bionic and platform API headers are generalized into a single |
| 27 | // `ndk_headers` rule. This rule has a `from` property that indicates a base |
| 28 | // directory from which headers are to be taken, and a `to` property that |
| 29 | // indicates where in the sysroot they should reside relative to usr/include. |
| 30 | // There is also a `srcs` property that is glob compatible for specifying which |
| 31 | // headers to include. |
| 32 | // |
| 33 | // Component 3: Stub Libraries |
| 34 | // The shared libraries in the NDK are not the actual shared libraries they |
| 35 | // refer to (to prevent people from accidentally loading them), but stub |
Joe Onorato | b4638c1 | 2021-10-27 15:47:06 -0700 | [diff] [blame] | 36 | // libraries with placeholder implementations of everything for use at build time |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 37 | // only. |
| 38 | // |
| 39 | // Since we don't actually need to know anything about the stub libraries aside |
| 40 | // from a list of functions and globals to be exposed, we can create these for |
| 41 | // every platform level in the current tree. This is handled by the |
| 42 | // ndk_library rule. |
| 43 | // |
| 44 | // Component 4: Static Libraries |
| 45 | // The NDK only provides static libraries for bionic, not the platform APIs. |
| 46 | // Since these need to be the actual implementation, we can't build old versions |
| 47 | // in the current platform tree. As such, legacy versions are checked in |
| 48 | // prebuilt to development/ndk, and a current version is built and archived as |
| 49 | // part of the platform build. The platfrom already builds these libraries, our |
| 50 | // NDK build rules only need to archive them for retrieval so they can be added |
| 51 | // to the prebuilts. |
| 52 | // |
| 53 | // TODO(danalbert): Write `ndk_static_library` rule. |
| 54 | |
| 55 | import ( |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 56 | "android/soong/android" |
| 57 | ) |
| 58 | |
| 59 | func init() { |
Colin Cross | 798bfce | 2016-10-12 14:28:16 -0700 | [diff] [blame] | 60 | android.RegisterModuleType("ndk_headers", ndkHeadersFactory) |
Jooyung Han | b90e491 | 2019-12-09 18:21:48 +0900 | [diff] [blame] | 61 | android.RegisterModuleType("ndk_library", NdkLibraryFactory) |
Dan Albert | 97f9c96 | 2018-05-24 15:02:16 -0700 | [diff] [blame] | 62 | android.RegisterModuleType("versioned_ndk_headers", versionedNdkHeadersFactory) |
Dan Albert | cb1b4b2 | 2018-05-24 15:06:11 -0700 | [diff] [blame] | 63 | android.RegisterModuleType("preprocessed_ndk_headers", preprocessedNdkHeadersFactory) |
Colin Cross | 798bfce | 2016-10-12 14:28:16 -0700 | [diff] [blame] | 64 | android.RegisterSingletonType("ndk", NdkSingleton) |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 65 | |
Colin Cross | cc0ce80 | 2019-04-02 16:14:11 -0700 | [diff] [blame] | 66 | pctx.Import("android/soong/android") |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 67 | } |
| 68 | |
Colin Cross | 70dda7e | 2019-10-01 22:05:35 -0700 | [diff] [blame] | 69 | func getNdkInstallBase(ctx android.PathContext) android.InstallPath { |
| 70 | return android.PathForNdkInstall(ctx) |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 71 | } |
| 72 | |
| 73 | // Returns the main install directory for the NDK sysroot. Usable with --sysroot. |
Colin Cross | 70dda7e | 2019-10-01 22:05:35 -0700 | [diff] [blame] | 74 | func getNdkSysrootBase(ctx android.PathContext) android.InstallPath { |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 75 | return getNdkInstallBase(ctx).Join(ctx, "sysroot") |
| 76 | } |
| 77 | |
Dan Albert | 6ab43d8 | 2017-12-13 15:05:04 -0800 | [diff] [blame] | 78 | // The base timestamp file depends on the NDK headers and stub shared libraries, |
| 79 | // but not the static libraries. This distinction is needed because the static |
| 80 | // libraries themselves might need to depend on the base sysroot. |
| 81 | func getNdkBaseTimestampFile(ctx android.PathContext) android.WritablePath { |
| 82 | return android.PathForOutput(ctx, "ndk_base.timestamp") |
| 83 | } |
| 84 | |
Chih-Hung Hsieh | 5b72153 | 2021-11-30 17:31:23 -0800 | [diff] [blame] | 85 | // The headers timestamp file depends only on the NDK headers. |
| 86 | // This is used mainly for .tidy files that do not need any stub libraries. |
| 87 | func getNdkHeadersTimestampFile(ctx android.PathContext) android.WritablePath { |
| 88 | return android.PathForOutput(ctx, "ndk_headers.timestamp") |
| 89 | } |
| 90 | |
Dan Albert | 6ab43d8 | 2017-12-13 15:05:04 -0800 | [diff] [blame] | 91 | // The full timestamp file depends on the base timestamp *and* the static |
| 92 | // libraries. |
| 93 | func getNdkFullTimestampFile(ctx android.PathContext) android.WritablePath { |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 94 | return android.PathForOutput(ctx, "ndk.timestamp") |
| 95 | } |
| 96 | |
Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 97 | func NdkSingleton() android.Singleton { |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 98 | return &ndkSingleton{} |
| 99 | } |
| 100 | |
| 101 | type ndkSingleton struct{} |
| 102 | |
Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 103 | func (n *ndkSingleton) GenerateBuildActions(ctx android.SingletonContext) { |
Dan Albert | 6ab43d8 | 2017-12-13 15:05:04 -0800 | [diff] [blame] | 104 | var staticLibInstallPaths android.Paths |
Chih-Hung Hsieh | 5b72153 | 2021-11-30 17:31:23 -0800 | [diff] [blame] | 105 | var headerPaths android.Paths |
Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 106 | var installPaths android.Paths |
| 107 | var licensePaths android.Paths |
| 108 | ctx.VisitAllModules(func(module android.Module) { |
Dan Willemsen | 95f4dbb | 2017-05-05 23:26:01 -0700 | [diff] [blame] | 109 | if m, ok := module.(android.Module); ok && !m.Enabled() { |
| 110 | return |
| 111 | } |
| 112 | |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 113 | if m, ok := module.(*headerModule); ok { |
Chih-Hung Hsieh | 5b72153 | 2021-11-30 17:31:23 -0800 | [diff] [blame] | 114 | headerPaths = append(headerPaths, m.installPaths...) |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 115 | installPaths = append(installPaths, m.installPaths...) |
Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 116 | licensePaths = append(licensePaths, m.licensePath) |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 117 | } |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 118 | |
Dan Albert | 97f9c96 | 2018-05-24 15:02:16 -0700 | [diff] [blame] | 119 | if m, ok := module.(*versionedHeaderModule); ok { |
Chih-Hung Hsieh | 5b72153 | 2021-11-30 17:31:23 -0800 | [diff] [blame] | 120 | headerPaths = append(headerPaths, m.installPaths...) |
Dan Albert | 269fab8 | 2017-02-15 17:31:33 -0800 | [diff] [blame] | 121 | installPaths = append(installPaths, m.installPaths...) |
Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 122 | licensePaths = append(licensePaths, m.licensePath) |
Dan Albert | 269fab8 | 2017-02-15 17:31:33 -0800 | [diff] [blame] | 123 | } |
| 124 | |
Dan Albert | cb1b4b2 | 2018-05-24 15:06:11 -0700 | [diff] [blame] | 125 | if m, ok := module.(*preprocessedHeadersModule); ok { |
Chih-Hung Hsieh | 5b72153 | 2021-11-30 17:31:23 -0800 | [diff] [blame] | 126 | headerPaths = append(headerPaths, m.installPaths...) |
Dan Albert | cb1b4b2 | 2018-05-24 15:06:11 -0700 | [diff] [blame] | 127 | installPaths = append(installPaths, m.installPaths...) |
| 128 | licensePaths = append(licensePaths, m.licensePath) |
| 129 | } |
| 130 | |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 131 | if m, ok := module.(*Module); ok { |
Colin Cross | 31076b3 | 2020-10-23 17:22:06 -0700 | [diff] [blame] | 132 | if installer, ok := m.installer.(*stubDecorator); ok && m.library.buildStubs() { |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 133 | installPaths = append(installPaths, installer.installPath) |
| 134 | } |
Dan Albert | f563d25 | 2017-10-13 00:29:00 -0700 | [diff] [blame] | 135 | |
| 136 | if library, ok := m.linker.(*libraryDecorator); ok { |
Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 137 | if library.ndkSysrootPath != nil { |
Dan Albert | 6ab43d8 | 2017-12-13 15:05:04 -0800 | [diff] [blame] | 138 | staticLibInstallPaths = append( |
| 139 | staticLibInstallPaths, library.ndkSysrootPath) |
Dan Albert | f563d25 | 2017-10-13 00:29:00 -0700 | [diff] [blame] | 140 | } |
| 141 | } |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 142 | } |
| 143 | }) |
| 144 | |
Ryan Prichard | b1c9d40 | 2018-08-20 22:06:01 -0700 | [diff] [blame] | 145 | // Include only a single copy of each license file. The Bionic NOTICE is |
| 146 | // long and is referenced by multiple Bionic modules. |
| 147 | licensePaths = android.FirstUniquePaths(licensePaths) |
| 148 | |
Dan Albert | c6345fb | 2016-10-20 01:36:11 -0700 | [diff] [blame] | 149 | combinedLicense := getNdkInstallBase(ctx).Join(ctx, "NOTICE") |
Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 150 | ctx.Build(pctx, android.BuildParams{ |
Colin Cross | 67a5c13 | 2017-05-09 13:45:28 -0700 | [diff] [blame] | 151 | Rule: android.Cat, |
| 152 | Description: "combine licenses", |
Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 153 | Output: combinedLicense, |
Colin Cross | 67a5c13 | 2017-05-09 13:45:28 -0700 | [diff] [blame] | 154 | Inputs: licensePaths, |
Dan Albert | c6345fb | 2016-10-20 01:36:11 -0700 | [diff] [blame] | 155 | }) |
| 156 | |
Dan Albert | 4922703 | 2021-06-15 13:25:25 -0700 | [diff] [blame] | 157 | baseDepPaths := append(installPaths, combinedLicense) |
Dan Albert | c6345fb | 2016-10-20 01:36:11 -0700 | [diff] [blame] | 158 | |
Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 159 | ctx.Build(pctx, android.BuildParams{ |
Dan Albert | 4922703 | 2021-06-15 13:25:25 -0700 | [diff] [blame] | 160 | Rule: android.Touch, |
| 161 | Output: getNdkBaseTimestampFile(ctx), |
| 162 | Implicits: baseDepPaths, |
| 163 | Validation: getNdkAbiDiffTimestampFile(ctx), |
Dan Albert | 6ab43d8 | 2017-12-13 15:05:04 -0800 | [diff] [blame] | 164 | }) |
| 165 | |
Chih-Hung Hsieh | 5b72153 | 2021-11-30 17:31:23 -0800 | [diff] [blame] | 166 | ctx.Build(pctx, android.BuildParams{ |
| 167 | Rule: android.Touch, |
| 168 | Output: getNdkHeadersTimestampFile(ctx), |
| 169 | Implicits: headerPaths, |
| 170 | }) |
| 171 | |
Dan Albert | 6ab43d8 | 2017-12-13 15:05:04 -0800 | [diff] [blame] | 172 | fullDepPaths := append(staticLibInstallPaths, getNdkBaseTimestampFile(ctx)) |
| 173 | |
Dan Albert | f1d14c7 | 2020-07-30 14:32:55 -0700 | [diff] [blame] | 174 | // There's a phony "ndk" rule defined in core/main.mk that depends on this. |
| 175 | // `m ndk` will build the sysroots for the architectures in the current |
| 176 | // lunch target. `build/soong/scripts/build-ndk-prebuilts.sh` will build the |
| 177 | // sysroots for all the NDK architectures and package them so they can be |
| 178 | // imported into the NDK's build. |
Dan Albert | 6ab43d8 | 2017-12-13 15:05:04 -0800 | [diff] [blame] | 179 | ctx.Build(pctx, android.BuildParams{ |
| 180 | Rule: android.Touch, |
| 181 | Output: getNdkFullTimestampFile(ctx), |
| 182 | Implicits: fullDepPaths, |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 183 | }) |
| 184 | } |