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 |
| 36 | // libraries with dummy implementations of everything for use at build time |
| 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 ( |
| 56 | "github.com/google/blueprint" |
| 57 | |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 58 | "android/soong/android" |
| 59 | ) |
| 60 | |
| 61 | func init() { |
Colin Cross | 798bfce | 2016-10-12 14:28:16 -0700 | [diff] [blame^] | 62 | android.RegisterModuleType("ndk_headers", ndkHeadersFactory) |
| 63 | android.RegisterModuleType("ndk_library", ndkLibraryFactory) |
| 64 | android.RegisterSingletonType("ndk", NdkSingleton) |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 65 | |
| 66 | pctx.Import("android/soong/common") |
| 67 | } |
| 68 | |
| 69 | func getNdkInstallBase(ctx android.ModuleContext) android.OutputPath { |
| 70 | return android.PathForOutput(ctx, "ndk") |
| 71 | } |
| 72 | |
| 73 | // Returns the main install directory for the NDK sysroot. Usable with --sysroot. |
| 74 | func getNdkSysrootBase(ctx android.ModuleContext) android.OutputPath { |
| 75 | return getNdkInstallBase(ctx).Join(ctx, "sysroot") |
| 76 | } |
| 77 | |
| 78 | func getNdkSysrootTimestampFile(ctx android.PathContext) android.Path { |
| 79 | return android.PathForOutput(ctx, "ndk.timestamp") |
| 80 | } |
| 81 | |
| 82 | func NdkSingleton() blueprint.Singleton { |
| 83 | return &ndkSingleton{} |
| 84 | } |
| 85 | |
| 86 | type ndkSingleton struct{} |
| 87 | |
| 88 | func (n *ndkSingleton) GenerateBuildActions(ctx blueprint.SingletonContext) { |
| 89 | installPaths := []string{} |
| 90 | ctx.VisitAllModules(func(module blueprint.Module) { |
| 91 | if m, ok := module.(*headerModule); ok { |
| 92 | installPaths = append(installPaths, m.installPaths...) |
| 93 | } |
| 94 | }) |
| 95 | |
| 96 | ctx.VisitAllModules(func(module blueprint.Module) { |
| 97 | if m, ok := module.(*Module); ok { |
Colin Cross | b916a38 | 2016-07-29 17:28:03 -0700 | [diff] [blame] | 98 | if installer, ok := m.installer.(*stubDecorator); ok { |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 99 | installPaths = append(installPaths, installer.installPath) |
| 100 | } |
| 101 | } |
| 102 | }) |
| 103 | |
| 104 | // There's a dummy "ndk" rule defined in ndk/Android.mk that depends on |
| 105 | // this. `m ndk` will build the sysroots. |
| 106 | ctx.Build(pctx, blueprint.BuildParams{ |
| 107 | Rule: android.Touch, |
| 108 | Outputs: []string{getNdkSysrootTimestampFile(ctx).String()}, |
| 109 | Implicits: installPaths, |
Colin Cross | f09c843 | 2016-09-16 12:51:30 -0700 | [diff] [blame] | 110 | Optional: true, |
Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 111 | }) |
| 112 | } |