| 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 | import ( | 
|  | 18 | "path/filepath" | 
|  | 19 |  | 
| Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 20 | "android/soong/android" | 
| Dan Albert | 266f991 | 2024-08-13 22:01:23 +0000 | [diff] [blame] | 21 |  | 
| Colin Cross | 8ff1058 | 2023-12-07 13:10:56 -0800 | [diff] [blame] | 22 | "github.com/google/blueprint" | 
| Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 23 | ) | 
|  | 24 |  | 
| Dan Albert | 269fab8 | 2017-02-15 17:31:33 -0800 | [diff] [blame] | 25 | var ( | 
| Dan Albert | cb1b4b2 | 2018-05-24 15:06:11 -0700 | [diff] [blame] | 26 | preprocessNdkHeader = pctx.AndroidStaticRule("preprocessNdkHeader", | 
|  | 27 | blueprint.RuleParams{ | 
|  | 28 | Command:     "$preprocessor -o $out $in", | 
|  | 29 | CommandDeps: []string{"$preprocessor"}, | 
|  | 30 | }, | 
|  | 31 | "preprocessor") | 
| Dan Albert | 269fab8 | 2017-02-15 17:31:33 -0800 | [diff] [blame] | 32 | ) | 
|  | 33 |  | 
| Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 34 | // Returns the NDK base include path for use with sdk_version current. Usable with -I. | 
| Dan Albert | 266f991 | 2024-08-13 22:01:23 +0000 | [diff] [blame] | 35 | func getCurrentIncludePath(ctx android.PathContext) android.OutputPath { | 
| Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 36 | return getNdkSysrootBase(ctx).Join(ctx, "usr/include") | 
|  | 37 | } | 
|  | 38 |  | 
| Dan Albert | 7122205 | 2018-05-24 15:00:05 -0700 | [diff] [blame] | 39 | type headerProperties struct { | 
| Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 40 | // Base directory of the headers being installed. As an example: | 
|  | 41 | // | 
|  | 42 | // ndk_headers { | 
|  | 43 | //     name: "foo", | 
|  | 44 | //     from: "include", | 
|  | 45 | //     to: "", | 
|  | 46 | //     srcs: ["include/foo/bar/baz.h"], | 
|  | 47 | // } | 
|  | 48 | // | 
|  | 49 | // Will install $SYSROOT/usr/include/foo/bar/baz.h. If `from` were instead | 
|  | 50 | // "include/foo", it would have installed $SYSROOT/usr/include/bar/baz.h. | 
| Nan Zhang | 0007d81 | 2017-11-07 10:57:05 -0800 | [diff] [blame] | 51 | From *string | 
| Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 52 |  | 
|  | 53 | // Install path within the sysroot. This is relative to usr/include. | 
| Nan Zhang | 0007d81 | 2017-11-07 10:57:05 -0800 | [diff] [blame] | 54 | To *string | 
| Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 55 |  | 
|  | 56 | // List of headers to install. Glob compatible. Common case is "include/**/*.h". | 
| Colin Cross | 27b922f | 2019-03-04 22:35:41 -0800 | [diff] [blame] | 57 | Srcs []string `android:"path"` | 
| Dan Albert | c6345fb | 2016-10-20 01:36:11 -0700 | [diff] [blame] | 58 |  | 
| Dan Albert | 19ff8b4 | 2018-05-24 15:00:48 -0700 | [diff] [blame] | 59 | // Source paths that should be excluded from the srcs glob. | 
| Colin Cross | 27b922f | 2019-03-04 22:35:41 -0800 | [diff] [blame] | 60 | Exclude_srcs []string `android:"path"` | 
| Dan Albert | 19ff8b4 | 2018-05-24 15:00:48 -0700 | [diff] [blame] | 61 |  | 
| Dan Albert | c6345fb | 2016-10-20 01:36:11 -0700 | [diff] [blame] | 62 | // Path to the NOTICE file associated with the headers. | 
| Colin Cross | 27b922f | 2019-03-04 22:35:41 -0800 | [diff] [blame] | 63 | License *string `android:"path"` | 
| Dan Albert | 266f991 | 2024-08-13 22:01:23 +0000 | [diff] [blame] | 64 |  | 
|  | 65 | // Set to true if the headers installed by this module should skip | 
|  | 66 | // verification. This step ensures that each header is self-contained (can | 
|  | 67 | // be #included alone) and is valid C. This should not be disabled except in | 
|  | 68 | // rare cases. Outside bionic and external, if you're using this option | 
|  | 69 | // you've probably made a mistake. | 
|  | 70 | Skip_verification *bool | 
| Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 71 | } | 
|  | 72 |  | 
|  | 73 | type headerModule struct { | 
|  | 74 | android.ModuleBase | 
|  | 75 |  | 
| Dan Albert | 7122205 | 2018-05-24 15:00:05 -0700 | [diff] [blame] | 76 | properties headerProperties | 
| Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 77 |  | 
| Aleksei Vetrov | 262ed1a | 2023-08-23 10:06:35 +0000 | [diff] [blame] | 78 | srcPaths     android.Paths | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 79 | installPaths android.Paths | 
| Colin Cross | 07e5161 | 2019-03-05 12:46:40 -0800 | [diff] [blame] | 80 | licensePath  android.Path | 
| Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 81 | } | 
|  | 82 |  | 
| Dan Albert | 269fab8 | 2017-02-15 17:31:33 -0800 | [diff] [blame] | 83 | func getHeaderInstallDir(ctx android.ModuleContext, header android.Path, from string, | 
| Spandan Das | f280b23 | 2024-04-04 21:25:51 +0000 | [diff] [blame] | 84 | to string) android.OutputPath { | 
| Dan Albert | 269fab8 | 2017-02-15 17:31:33 -0800 | [diff] [blame] | 85 | // Output path is the sysroot base + "usr/include" + to directory + directory component | 
|  | 86 | // of the file without the leading from directory stripped. | 
|  | 87 | // | 
|  | 88 | // Given: | 
|  | 89 | // sysroot base = "ndk/sysroot" | 
|  | 90 | // from = "include/foo" | 
|  | 91 | // to = "bar" | 
|  | 92 | // header = "include/foo/woodly/doodly.h" | 
|  | 93 | // output path = "ndk/sysroot/usr/include/bar/woodly/doodly.h" | 
|  | 94 |  | 
|  | 95 | // full/platform/path/to/include/foo | 
|  | 96 | fullFromPath := android.PathForModuleSrc(ctx, from) | 
|  | 97 |  | 
|  | 98 | // full/platform/path/to/include/foo/woodly | 
|  | 99 | headerDir := filepath.Dir(header.String()) | 
|  | 100 |  | 
|  | 101 | // woodly | 
|  | 102 | strippedHeaderDir, err := filepath.Rel(fullFromPath.String(), headerDir) | 
|  | 103 | if err != nil { | 
|  | 104 | ctx.ModuleErrorf("filepath.Rel(%q, %q) failed: %s", headerDir, | 
|  | 105 | fullFromPath.String(), err) | 
|  | 106 | } | 
|  | 107 |  | 
|  | 108 | // full/platform/path/to/sysroot/usr/include/bar/woodly | 
|  | 109 | installDir := getCurrentIncludePath(ctx).Join(ctx, to, strippedHeaderDir) | 
|  | 110 |  | 
|  | 111 | // full/platform/path/to/sysroot/usr/include/bar/woodly/doodly.h | 
|  | 112 | return installDir | 
|  | 113 | } | 
|  | 114 |  | 
| Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 115 | func (m *headerModule) GenerateAndroidBuildActions(ctx android.ModuleContext) { | 
| Nan Zhang | 0007d81 | 2017-11-07 10:57:05 -0800 | [diff] [blame] | 116 | if String(m.properties.License) == "" { | 
| Dan Albert | c6345fb | 2016-10-20 01:36:11 -0700 | [diff] [blame] | 117 | ctx.PropertyErrorf("license", "field is required") | 
|  | 118 | } | 
|  | 119 |  | 
| Nan Zhang | 0007d81 | 2017-11-07 10:57:05 -0800 | [diff] [blame] | 120 | m.licensePath = android.PathForModuleSrc(ctx, String(m.properties.License)) | 
| Dan Albert | c6345fb | 2016-10-20 01:36:11 -0700 | [diff] [blame] | 121 |  | 
| Aleksei Vetrov | 262ed1a | 2023-08-23 10:06:35 +0000 | [diff] [blame] | 122 | m.srcPaths = android.PathsForModuleSrcExcludes(ctx, m.properties.Srcs, m.properties.Exclude_srcs) | 
|  | 123 | for _, header := range m.srcPaths { | 
| Nan Zhang | 0007d81 | 2017-11-07 10:57:05 -0800 | [diff] [blame] | 124 | installDir := getHeaderInstallDir(ctx, header, String(m.properties.From), | 
|  | 125 | String(m.properties.To)) | 
| Dan Albert | 269fab8 | 2017-02-15 17:31:33 -0800 | [diff] [blame] | 126 | installPath := installDir.Join(ctx, header.Base()) | 
| Spandan Das | f280b23 | 2024-04-04 21:25:51 +0000 | [diff] [blame] | 127 | ctx.Build(pctx, android.BuildParams{ | 
|  | 128 | Rule:   android.Cp, | 
|  | 129 | Input:  header, | 
|  | 130 | Output: installPath, | 
|  | 131 | }) | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 132 | m.installPaths = append(m.installPaths, installPath) | 
| Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 133 | } | 
|  | 134 |  | 
|  | 135 | if len(m.installPaths) == 0 { | 
|  | 136 | ctx.ModuleErrorf("srcs %q matched zero files", m.properties.Srcs) | 
|  | 137 | } | 
|  | 138 | } | 
|  | 139 |  | 
| Patrice Arruda | 6ea4211 | 2019-04-03 08:43:30 -0700 | [diff] [blame] | 140 | // ndk_headers installs the sets of ndk headers defined in the srcs property | 
|  | 141 | // to the sysroot base + "usr/include" + to directory + directory component. | 
|  | 142 | // ndk_headers requires the license file to be specified. Example: | 
|  | 143 | // | 
| Colin Cross | d079e0b | 2022-08-16 10:27:33 -0700 | [diff] [blame] | 144 | //	Given: | 
|  | 145 | //	sysroot base = "ndk/sysroot" | 
|  | 146 | //	from = "include/foo" | 
|  | 147 | //	to = "bar" | 
|  | 148 | //	header = "include/foo/woodly/doodly.h" | 
|  | 149 | //	output path = "ndk/sysroot/usr/include/bar/woodly/doodly.h" | 
| Spandan Das | 319711b | 2023-09-19 19:04:41 +0000 | [diff] [blame] | 150 | func NdkHeadersFactory() android.Module { | 
| Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 151 | module := &headerModule{} | 
| Colin Cross | 3624285 | 2017-06-23 15:06:31 -0700 | [diff] [blame] | 152 | module.AddProperties(&module.properties) | 
|  | 153 | android.InitAndroidModule(module) | 
|  | 154 | return module | 
| Dan Albert | 914449f | 2016-06-17 16:45:24 -0700 | [diff] [blame] | 155 | } | 
| Dan Albert | 269fab8 | 2017-02-15 17:31:33 -0800 | [diff] [blame] | 156 |  | 
| Spandan Das | 0773a60 | 2022-08-16 00:55:11 +0000 | [diff] [blame] | 157 | // preprocessed_ndk_header { | 
|  | 158 | // | 
|  | 159 | //	name: "foo", | 
|  | 160 | //	preprocessor: "foo.sh", | 
|  | 161 | //	srcs: [...], | 
|  | 162 | //	to: "android", | 
|  | 163 | // | 
|  | 164 | // } | 
| Dan Albert | cb1b4b2 | 2018-05-24 15:06:11 -0700 | [diff] [blame] | 165 | // | 
|  | 166 | // Will invoke the preprocessor as: | 
| Colin Cross | d079e0b | 2022-08-16 10:27:33 -0700 | [diff] [blame] | 167 | // | 
|  | 168 | //	$preprocessor -o $SYSROOT/usr/include/android/needs_preproc.h $src | 
|  | 169 | // | 
| Dan Albert | cb1b4b2 | 2018-05-24 15:06:11 -0700 | [diff] [blame] | 170 | // For each src in srcs. | 
|  | 171 | type preprocessedHeadersProperties struct { | 
|  | 172 | // The preprocessor to run. Must be a program inside the source directory | 
|  | 173 | // with no dependencies. | 
|  | 174 | Preprocessor *string | 
|  | 175 |  | 
|  | 176 | // Source path to the files to be preprocessed. | 
|  | 177 | Srcs []string | 
|  | 178 |  | 
|  | 179 | // Source paths that should be excluded from the srcs glob. | 
|  | 180 | Exclude_srcs []string | 
|  | 181 |  | 
|  | 182 | // Install path within the sysroot. This is relative to usr/include. | 
|  | 183 | To *string | 
|  | 184 |  | 
|  | 185 | // Path to the NOTICE file associated with the headers. | 
|  | 186 | License *string | 
| Dan Albert | 266f991 | 2024-08-13 22:01:23 +0000 | [diff] [blame] | 187 |  | 
|  | 188 | // Set to true if the headers installed by this module should skip | 
|  | 189 | // verification. This step ensures that each header is self-contained (can | 
|  | 190 | // be #included alone) and is valid C. This should not be disabled except in | 
|  | 191 | // rare cases. Outside bionic and external, if you're using this option | 
|  | 192 | // you've probably made a mistake. | 
|  | 193 | Skip_verification *bool | 
| Dan Albert | cb1b4b2 | 2018-05-24 15:06:11 -0700 | [diff] [blame] | 194 | } | 
|  | 195 |  | 
|  | 196 | type preprocessedHeadersModule struct { | 
|  | 197 | android.ModuleBase | 
|  | 198 |  | 
|  | 199 | properties preprocessedHeadersProperties | 
|  | 200 |  | 
| Aleksei Vetrov | 262ed1a | 2023-08-23 10:06:35 +0000 | [diff] [blame] | 201 | srcPaths     android.Paths | 
| Dan Albert | cb1b4b2 | 2018-05-24 15:06:11 -0700 | [diff] [blame] | 202 | installPaths android.Paths | 
| Colin Cross | 07e5161 | 2019-03-05 12:46:40 -0800 | [diff] [blame] | 203 | licensePath  android.Path | 
| Dan Albert | cb1b4b2 | 2018-05-24 15:06:11 -0700 | [diff] [blame] | 204 | } | 
|  | 205 |  | 
| Dan Albert | cb1b4b2 | 2018-05-24 15:06:11 -0700 | [diff] [blame] | 206 | func (m *preprocessedHeadersModule) GenerateAndroidBuildActions(ctx android.ModuleContext) { | 
|  | 207 | if String(m.properties.License) == "" { | 
|  | 208 | ctx.PropertyErrorf("license", "field is required") | 
|  | 209 | } | 
|  | 210 |  | 
|  | 211 | preprocessor := android.PathForModuleSrc(ctx, String(m.properties.Preprocessor)) | 
|  | 212 | m.licensePath = android.PathForModuleSrc(ctx, String(m.properties.License)) | 
|  | 213 |  | 
| Aleksei Vetrov | 262ed1a | 2023-08-23 10:06:35 +0000 | [diff] [blame] | 214 | m.srcPaths = android.PathsForModuleSrcExcludes(ctx, m.properties.Srcs, m.properties.Exclude_srcs) | 
| Dan Albert | cb1b4b2 | 2018-05-24 15:06:11 -0700 | [diff] [blame] | 215 | installDir := getCurrentIncludePath(ctx).Join(ctx, String(m.properties.To)) | 
| Aleksei Vetrov | 262ed1a | 2023-08-23 10:06:35 +0000 | [diff] [blame] | 216 | for _, src := range m.srcPaths { | 
| Dan Albert | cb1b4b2 | 2018-05-24 15:06:11 -0700 | [diff] [blame] | 217 | installPath := installDir.Join(ctx, src.Base()) | 
|  | 218 | m.installPaths = append(m.installPaths, installPath) | 
|  | 219 |  | 
|  | 220 | ctx.Build(pctx, android.BuildParams{ | 
|  | 221 | Rule:        preprocessNdkHeader, | 
|  | 222 | Description: "preprocess " + src.Rel(), | 
|  | 223 | Input:       src, | 
|  | 224 | Output:      installPath, | 
|  | 225 | Args: map[string]string{ | 
|  | 226 | "preprocessor": preprocessor.String(), | 
|  | 227 | }, | 
|  | 228 | }) | 
|  | 229 | } | 
|  | 230 |  | 
|  | 231 | if len(m.installPaths) == 0 { | 
|  | 232 | ctx.ModuleErrorf("srcs %q matched zero files", m.properties.Srcs) | 
|  | 233 | } | 
|  | 234 | } | 
|  | 235 |  | 
| Patrice Arruda | 6ea4211 | 2019-04-03 08:43:30 -0700 | [diff] [blame] | 236 | // preprocessed_ndk_headers preprocesses all the ndk headers listed in the srcs | 
|  | 237 | // property by executing the command defined in the preprocessor property. | 
| Dan Albert | cb1b4b2 | 2018-05-24 15:06:11 -0700 | [diff] [blame] | 238 | func preprocessedNdkHeadersFactory() android.Module { | 
|  | 239 | module := &preprocessedHeadersModule{} | 
|  | 240 |  | 
|  | 241 | module.AddProperties(&module.properties) | 
|  | 242 |  | 
| Dan Willemsen | 4f644da | 2018-10-10 17:18:08 -0700 | [diff] [blame] | 243 | android.InitAndroidModule(module) | 
| Dan Albert | cb1b4b2 | 2018-05-24 15:06:11 -0700 | [diff] [blame] | 244 |  | 
|  | 245 | return module | 
|  | 246 | } |