blob: ee11db1016fd68b7b9ce6949bb2c2b0a2a55868c [file] [log] [blame]
Dan Albert914449f2016-06-17 16:45:24 -07001// 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
15package 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 Onoratob4638c12021-10-27 15:47:06 -070036// libraries with placeholder implementations of everything for use at build time
Dan Albert914449f2016-06-17 16:45:24 -070037// 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
55import (
Dan Albert914449f2016-06-17 16:45:24 -070056 "android/soong/android"
57)
58
59func init() {
Colin Cross798bfce2016-10-12 14:28:16 -070060 android.RegisterModuleType("ndk_headers", ndkHeadersFactory)
Jooyung Hanb90e4912019-12-09 18:21:48 +090061 android.RegisterModuleType("ndk_library", NdkLibraryFactory)
Dan Albert97f9c962018-05-24 15:02:16 -070062 android.RegisterModuleType("versioned_ndk_headers", versionedNdkHeadersFactory)
Dan Albertcb1b4b22018-05-24 15:06:11 -070063 android.RegisterModuleType("preprocessed_ndk_headers", preprocessedNdkHeadersFactory)
Colin Cross798bfce2016-10-12 14:28:16 -070064 android.RegisterSingletonType("ndk", NdkSingleton)
Dan Albert914449f2016-06-17 16:45:24 -070065
Colin Crosscc0ce802019-04-02 16:14:11 -070066 pctx.Import("android/soong/android")
Dan Albert914449f2016-06-17 16:45:24 -070067}
68
Colin Cross70dda7e2019-10-01 22:05:35 -070069func getNdkInstallBase(ctx android.PathContext) android.InstallPath {
70 return android.PathForNdkInstall(ctx)
Dan Albert914449f2016-06-17 16:45:24 -070071}
72
73// Returns the main install directory for the NDK sysroot. Usable with --sysroot.
Colin Cross70dda7e2019-10-01 22:05:35 -070074func getNdkSysrootBase(ctx android.PathContext) android.InstallPath {
Dan Albert914449f2016-06-17 16:45:24 -070075 return getNdkInstallBase(ctx).Join(ctx, "sysroot")
76}
77
Dan Albert6ab43d82017-12-13 15:05:04 -080078// 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.
81func getNdkBaseTimestampFile(ctx android.PathContext) android.WritablePath {
82 return android.PathForOutput(ctx, "ndk_base.timestamp")
83}
84
Chih-Hung Hsieh5b721532021-11-30 17:31:23 -080085// 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.
87func getNdkHeadersTimestampFile(ctx android.PathContext) android.WritablePath {
88 return android.PathForOutput(ctx, "ndk_headers.timestamp")
89}
90
Dan Albert6ab43d82017-12-13 15:05:04 -080091// The full timestamp file depends on the base timestamp *and* the static
92// libraries.
93func getNdkFullTimestampFile(ctx android.PathContext) android.WritablePath {
Dan Albert914449f2016-06-17 16:45:24 -070094 return android.PathForOutput(ctx, "ndk.timestamp")
95}
96
Chih-Hung Hsieh5b721532021-11-30 17:31:23 -080097// Replace ndk_base.timestamp and ndk.timestamp with ndk_headers.timestamp.
98func skipNdkLibraryDeps(ctx android.ModuleContext, paths android.Paths) android.Paths {
99 var newPaths android.Paths
100 baseTimestamp := getNdkBaseTimestampFile(ctx)
101 fullTimestamp := getNdkFullTimestampFile(ctx)
102 headersTimestamp := getNdkHeadersTimestampFile(ctx)
103 for _, path := range paths {
104 if path == baseTimestamp || path == fullTimestamp {
105 path = headersTimestamp
106 }
107 newPaths = append(newPaths, path)
108 }
109 return newPaths
110}
111
Colin Cross0875c522017-11-28 17:34:01 -0800112func NdkSingleton() android.Singleton {
Dan Albert914449f2016-06-17 16:45:24 -0700113 return &ndkSingleton{}
114}
115
116type ndkSingleton struct{}
117
Colin Cross0875c522017-11-28 17:34:01 -0800118func (n *ndkSingleton) GenerateBuildActions(ctx android.SingletonContext) {
Dan Albert6ab43d82017-12-13 15:05:04 -0800119 var staticLibInstallPaths android.Paths
Chih-Hung Hsieh5b721532021-11-30 17:31:23 -0800120 var headerPaths android.Paths
Colin Cross0875c522017-11-28 17:34:01 -0800121 var installPaths android.Paths
122 var licensePaths android.Paths
123 ctx.VisitAllModules(func(module android.Module) {
Dan Willemsen95f4dbb2017-05-05 23:26:01 -0700124 if m, ok := module.(android.Module); ok && !m.Enabled() {
125 return
126 }
127
Dan Albert914449f2016-06-17 16:45:24 -0700128 if m, ok := module.(*headerModule); ok {
Chih-Hung Hsieh5b721532021-11-30 17:31:23 -0800129 headerPaths = append(headerPaths, m.installPaths...)
Dan Albert914449f2016-06-17 16:45:24 -0700130 installPaths = append(installPaths, m.installPaths...)
Colin Cross0875c522017-11-28 17:34:01 -0800131 licensePaths = append(licensePaths, m.licensePath)
Dan Albert914449f2016-06-17 16:45:24 -0700132 }
Dan Albert914449f2016-06-17 16:45:24 -0700133
Dan Albert97f9c962018-05-24 15:02:16 -0700134 if m, ok := module.(*versionedHeaderModule); ok {
Chih-Hung Hsieh5b721532021-11-30 17:31:23 -0800135 headerPaths = append(headerPaths, m.installPaths...)
Dan Albert269fab82017-02-15 17:31:33 -0800136 installPaths = append(installPaths, m.installPaths...)
Colin Cross0875c522017-11-28 17:34:01 -0800137 licensePaths = append(licensePaths, m.licensePath)
Dan Albert269fab82017-02-15 17:31:33 -0800138 }
139
Dan Albertcb1b4b22018-05-24 15:06:11 -0700140 if m, ok := module.(*preprocessedHeadersModule); ok {
Chih-Hung Hsieh5b721532021-11-30 17:31:23 -0800141 headerPaths = append(headerPaths, m.installPaths...)
Dan Albertcb1b4b22018-05-24 15:06:11 -0700142 installPaths = append(installPaths, m.installPaths...)
143 licensePaths = append(licensePaths, m.licensePath)
144 }
145
Dan Albert914449f2016-06-17 16:45:24 -0700146 if m, ok := module.(*Module); ok {
Colin Cross31076b32020-10-23 17:22:06 -0700147 if installer, ok := m.installer.(*stubDecorator); ok && m.library.buildStubs() {
Dan Albert914449f2016-06-17 16:45:24 -0700148 installPaths = append(installPaths, installer.installPath)
149 }
Dan Albertf563d252017-10-13 00:29:00 -0700150
151 if library, ok := m.linker.(*libraryDecorator); ok {
Colin Cross0875c522017-11-28 17:34:01 -0800152 if library.ndkSysrootPath != nil {
Dan Albert6ab43d82017-12-13 15:05:04 -0800153 staticLibInstallPaths = append(
154 staticLibInstallPaths, library.ndkSysrootPath)
Dan Albertf563d252017-10-13 00:29:00 -0700155 }
156 }
Dan Albert914449f2016-06-17 16:45:24 -0700157 }
158 })
159
Ryan Prichardb1c9d402018-08-20 22:06:01 -0700160 // Include only a single copy of each license file. The Bionic NOTICE is
161 // long and is referenced by multiple Bionic modules.
162 licensePaths = android.FirstUniquePaths(licensePaths)
163
Dan Albertc6345fb2016-10-20 01:36:11 -0700164 combinedLicense := getNdkInstallBase(ctx).Join(ctx, "NOTICE")
Colin Cross0875c522017-11-28 17:34:01 -0800165 ctx.Build(pctx, android.BuildParams{
Colin Cross67a5c132017-05-09 13:45:28 -0700166 Rule: android.Cat,
167 Description: "combine licenses",
Colin Cross0875c522017-11-28 17:34:01 -0800168 Output: combinedLicense,
Colin Cross67a5c132017-05-09 13:45:28 -0700169 Inputs: licensePaths,
Dan Albertc6345fb2016-10-20 01:36:11 -0700170 })
171
Dan Albert49227032021-06-15 13:25:25 -0700172 baseDepPaths := append(installPaths, combinedLicense)
Dan Albertc6345fb2016-10-20 01:36:11 -0700173
Colin Cross0875c522017-11-28 17:34:01 -0800174 ctx.Build(pctx, android.BuildParams{
Dan Albert49227032021-06-15 13:25:25 -0700175 Rule: android.Touch,
176 Output: getNdkBaseTimestampFile(ctx),
177 Implicits: baseDepPaths,
178 Validation: getNdkAbiDiffTimestampFile(ctx),
Dan Albert6ab43d82017-12-13 15:05:04 -0800179 })
180
Chih-Hung Hsieh5b721532021-11-30 17:31:23 -0800181 ctx.Build(pctx, android.BuildParams{
182 Rule: android.Touch,
183 Output: getNdkHeadersTimestampFile(ctx),
184 Implicits: headerPaths,
185 })
186
Dan Albert6ab43d82017-12-13 15:05:04 -0800187 fullDepPaths := append(staticLibInstallPaths, getNdkBaseTimestampFile(ctx))
188
Dan Albertf1d14c72020-07-30 14:32:55 -0700189 // There's a phony "ndk" rule defined in core/main.mk that depends on this.
190 // `m ndk` will build the sysroots for the architectures in the current
191 // lunch target. `build/soong/scripts/build-ndk-prebuilts.sh` will build the
192 // sysroots for all the NDK architectures and package them so they can be
193 // imported into the NDK's build.
Dan Albert6ab43d82017-12-13 15:05:04 -0800194 ctx.Build(pctx, android.BuildParams{
195 Rule: android.Touch,
196 Output: getNdkFullTimestampFile(ctx),
197 Implicits: fullDepPaths,
Dan Albert914449f2016-06-17 16:45:24 -0700198 })
199}