Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 1 | // Copyright 2019 The Android Open Source Project |
| 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 rust |
| 16 | |
| 17 | import ( |
| 18 | "fmt" |
| 19 | "path/filepath" |
| 20 | |
Ivan Lozano | ad8b18b | 2019-10-31 19:38:29 -0700 | [diff] [blame] | 21 | "github.com/google/blueprint/proptools" |
| 22 | |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 23 | "android/soong/android" |
| 24 | "android/soong/rust/config" |
| 25 | ) |
| 26 | |
Thiébaud Weksteen | e81c924 | 2020-08-03 10:46:28 +0200 | [diff] [blame] | 27 | func (compiler *baseCompiler) edition() string { |
Chih-Hung Hsieh | 961a30c | 2019-10-03 09:47:06 -0700 | [diff] [blame] | 28 | return proptools.StringDefault(compiler.Properties.Edition, config.DefaultEdition) |
| 29 | } |
| 30 | |
Matthew Maurer | 99020b0 | 2019-10-31 10:44:40 -0700 | [diff] [blame] | 31 | func (compiler *baseCompiler) setNoStdlibs() { |
| 32 | compiler.Properties.No_stdlibs = proptools.BoolPtr(true) |
| 33 | } |
| 34 | |
Thiébaud Weksteen | 9e8451e | 2020-08-13 12:55:59 +0200 | [diff] [blame] | 35 | func (compiler *baseCompiler) disableLints() { |
| 36 | compiler.Properties.Lints = proptools.StringPtr("none") |
Stephen Crane | da931d4 | 2020-08-04 13:02:28 -0700 | [diff] [blame] | 37 | } |
| 38 | |
Chih-Hung Hsieh | 9a4a7ba | 2019-12-12 19:36:05 -0800 | [diff] [blame] | 39 | func NewBaseCompiler(dir, dir64 string, location installLocation) *baseCompiler { |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 40 | return &baseCompiler{ |
Chih-Hung Hsieh | 961a30c | 2019-10-03 09:47:06 -0700 | [diff] [blame] | 41 | Properties: BaseCompilerProperties{}, |
| 42 | dir: dir, |
| 43 | dir64: dir64, |
Chih-Hung Hsieh | 9a4a7ba | 2019-12-12 19:36:05 -0800 | [diff] [blame] | 44 | location: location, |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 45 | } |
| 46 | } |
| 47 | |
Chih-Hung Hsieh | 9a4a7ba | 2019-12-12 19:36:05 -0800 | [diff] [blame] | 48 | type installLocation int |
| 49 | |
| 50 | const ( |
| 51 | InstallInSystem installLocation = 0 |
| 52 | InstallInData = iota |
Ivan Lozano | 4384568 | 2020-07-09 21:03:28 -0400 | [diff] [blame] | 53 | |
| 54 | incorrectSourcesError = "srcs can only contain one path for a rust file and source providers prefixed by \":\"" |
Chih-Hung Hsieh | 9a4a7ba | 2019-12-12 19:36:05 -0800 | [diff] [blame] | 55 | ) |
| 56 | |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 57 | type BaseCompilerProperties struct { |
Ivan Lozano | 8a23fa4 | 2020-06-16 10:26:57 -0400 | [diff] [blame] | 58 | // path to the source file that is the main entry point of the program (e.g. main.rs or lib.rs) |
| 59 | Srcs []string `android:"path,arch_variant"` |
| 60 | |
Thiébaud Weksteen | 9e8451e | 2020-08-13 12:55:59 +0200 | [diff] [blame] | 61 | // name of the lint set that should be used to validate this module. |
| 62 | // |
| 63 | // Possible values are "default" (for using a sensible set of lints |
| 64 | // depending on the module's location), "android" (for the strictest |
| 65 | // lint set that applies to all Android platform code), "vendor" (for |
| 66 | // a relaxed set) and "none" (for ignoring all lint warnings and |
| 67 | // errors). The default value is "default". |
| 68 | Lints *string |
Chih-Hung Hsieh | efdd7ac | 2019-09-26 18:59:27 -0700 | [diff] [blame] | 69 | |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 70 | // flags to pass to rustc |
| 71 | Flags []string `android:"path,arch_variant"` |
| 72 | |
| 73 | // flags to pass to the linker |
| 74 | Ld_flags []string `android:"path,arch_variant"` |
| 75 | |
| 76 | // list of rust rlib crate dependencies |
| 77 | Rlibs []string `android:"arch_variant"` |
| 78 | |
| 79 | // list of rust dylib crate dependencies |
| 80 | Dylibs []string `android:"arch_variant"` |
| 81 | |
Matthew Maurer | 0f003b1 | 2020-06-29 14:34:06 -0700 | [diff] [blame] | 82 | // list of rust automatic crate dependencies |
| 83 | Rustlibs []string `android:"arch_variant"` |
| 84 | |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 85 | // list of rust proc_macro crate dependencies |
| 86 | Proc_macros []string `android:"arch_variant"` |
| 87 | |
| 88 | // list of C shared library dependencies |
| 89 | Shared_libs []string `android:"arch_variant"` |
| 90 | |
| 91 | // list of C static library dependencies |
| 92 | Static_libs []string `android:"arch_variant"` |
| 93 | |
Ivan Lozano | 26ecd6c | 2020-07-31 13:40:31 -0400 | [diff] [blame] | 94 | // crate name, required for modules which produce Rust libraries: rust_library, rust_ffi and SourceProvider |
| 95 | // modules which create library variants (rust_bindgen). This must be the expected extern crate name used in |
| 96 | // source, and is required to conform to an enforced format matching library output files (if the output file is |
| 97 | // lib<someName><suffix>, the crate_name property must be <someName>). |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 98 | Crate_name string `android:"arch_variant"` |
| 99 | |
| 100 | // list of features to enable for this crate |
| 101 | Features []string `android:"arch_variant"` |
| 102 | |
| 103 | // specific rust edition that should be used if the default version is not desired |
| 104 | Edition *string `android:"arch_variant"` |
| 105 | |
| 106 | // sets name of the output |
| 107 | Stem *string `android:"arch_variant"` |
| 108 | |
| 109 | // append to name of output |
| 110 | Suffix *string `android:"arch_variant"` |
| 111 | |
| 112 | // install to a subdirectory of the default install path for the module |
| 113 | Relative_install_path *string `android:"arch_variant"` |
Matthew Maurer | 99020b0 | 2019-10-31 10:44:40 -0700 | [diff] [blame] | 114 | |
| 115 | // whether to suppress inclusion of standard crates - defaults to false |
| 116 | No_stdlibs *bool |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 117 | } |
| 118 | |
| 119 | type baseCompiler struct { |
Ivan Lozano | 8a23fa4 | 2020-06-16 10:26:57 -0400 | [diff] [blame] | 120 | Properties BaseCompilerProperties |
Ivan Lozano | 8a23fa4 | 2020-06-16 10:26:57 -0400 | [diff] [blame] | 121 | coverageFile android.Path //rustc generates a single gcno file |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 122 | |
| 123 | // Install related |
| 124 | dir string |
| 125 | dir64 string |
| 126 | subDir string |
| 127 | relative string |
Colin Cross | 70dda7e | 2019-10-01 22:05:35 -0700 | [diff] [blame] | 128 | path android.InstallPath |
Chih-Hung Hsieh | 9a4a7ba | 2019-12-12 19:36:05 -0800 | [diff] [blame] | 129 | location installLocation |
Ivan Lozano | 8a23fa4 | 2020-06-16 10:26:57 -0400 | [diff] [blame] | 130 | |
| 131 | coverageOutputZipFile android.OptionalPath |
Ivan Lozano | 8a23fa4 | 2020-06-16 10:26:57 -0400 | [diff] [blame] | 132 | distFile android.OptionalPath |
Thiébaud Weksteen | fabaff6 | 2020-08-27 13:48:36 +0200 | [diff] [blame] | 133 | // Stripped output file. If Valid(), this file will be installed instead of outputFile. |
| 134 | strippedOutputFile android.OptionalPath |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 135 | } |
| 136 | |
Ivan Lozano | 26ecd6c | 2020-07-31 13:40:31 -0400 | [diff] [blame] | 137 | func (compiler *baseCompiler) Disabled() bool { |
| 138 | return false |
| 139 | } |
| 140 | |
| 141 | func (compiler *baseCompiler) SetDisabled() { |
| 142 | panic("baseCompiler does not implement SetDisabled()") |
| 143 | } |
| 144 | |
Ivan Lozano | a0cd8f9 | 2020-04-09 09:56:02 -0400 | [diff] [blame] | 145 | func (compiler *baseCompiler) coverageOutputZipPath() android.OptionalPath { |
| 146 | panic("baseCompiler does not implement coverageOutputZipPath()") |
| 147 | } |
| 148 | |
Ivan Lozano | 042504f | 2020-08-18 14:31:23 -0400 | [diff] [blame] | 149 | func (compiler *baseCompiler) static() bool { |
| 150 | return false |
| 151 | } |
| 152 | |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 153 | var _ compiler = (*baseCompiler)(nil) |
| 154 | |
Chih-Hung Hsieh | 9a4a7ba | 2019-12-12 19:36:05 -0800 | [diff] [blame] | 155 | func (compiler *baseCompiler) inData() bool { |
| 156 | return compiler.location == InstallInData |
| 157 | } |
| 158 | |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 159 | func (compiler *baseCompiler) compilerProps() []interface{} { |
| 160 | return []interface{}{&compiler.Properties} |
| 161 | } |
| 162 | |
| 163 | func (compiler *baseCompiler) featuresToFlags(features []string) []string { |
| 164 | flags := []string{} |
| 165 | for _, feature := range features { |
| 166 | flags = append(flags, "--cfg 'feature=\""+feature+"\"'") |
| 167 | } |
| 168 | return flags |
| 169 | } |
| 170 | |
| 171 | func (compiler *baseCompiler) compilerFlags(ctx ModuleContext, flags Flags) Flags { |
| 172 | |
Thiébaud Weksteen | 9e8451e | 2020-08-13 12:55:59 +0200 | [diff] [blame] | 173 | lintFlags, err := config.RustcLintsForDir(ctx.ModuleDir(), compiler.Properties.Lints) |
| 174 | if err != nil { |
| 175 | ctx.PropertyErrorf("lints", err.Error()) |
Chih-Hung Hsieh | efdd7ac | 2019-09-26 18:59:27 -0700 | [diff] [blame] | 176 | } |
Thiébaud Weksteen | 9e8451e | 2020-08-13 12:55:59 +0200 | [diff] [blame] | 177 | flags.RustFlags = append(flags.RustFlags, lintFlags) |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 178 | flags.RustFlags = append(flags.RustFlags, compiler.Properties.Flags...) |
| 179 | flags.RustFlags = append(flags.RustFlags, compiler.featuresToFlags(compiler.Properties.Features)...) |
Thiébaud Weksteen | e81c924 | 2020-08-03 10:46:28 +0200 | [diff] [blame] | 180 | flags.RustFlags = append(flags.RustFlags, "--edition="+compiler.edition()) |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 181 | flags.LinkFlags = append(flags.LinkFlags, compiler.Properties.Ld_flags...) |
Joel Galenson | 724286c | 2019-09-30 13:01:37 -0700 | [diff] [blame] | 182 | flags.GlobalRustFlags = append(flags.GlobalRustFlags, config.GlobalRustFlags...) |
Ivan Lozano | f1c8433 | 2019-09-20 11:00:37 -0700 | [diff] [blame] | 183 | flags.GlobalRustFlags = append(flags.GlobalRustFlags, ctx.toolchain().ToolchainRustFlags()) |
| 184 | flags.GlobalLinkFlags = append(flags.GlobalLinkFlags, ctx.toolchain().ToolchainLinkFlags()) |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 185 | |
| 186 | if ctx.Host() && !ctx.Windows() { |
| 187 | rpath_prefix := `\$$ORIGIN/` |
| 188 | if ctx.Darwin() { |
| 189 | rpath_prefix = "@loader_path/" |
| 190 | } |
| 191 | |
| 192 | var rpath string |
| 193 | if ctx.toolchain().Is64Bit() { |
| 194 | rpath = "lib64" |
| 195 | } else { |
| 196 | rpath = "lib" |
| 197 | } |
| 198 | flags.LinkFlags = append(flags.LinkFlags, "-Wl,-rpath,"+rpath_prefix+rpath) |
| 199 | flags.LinkFlags = append(flags.LinkFlags, "-Wl,-rpath,"+rpath_prefix+"../"+rpath) |
| 200 | } |
| 201 | |
| 202 | return flags |
| 203 | } |
| 204 | |
| 205 | func (compiler *baseCompiler) compile(ctx ModuleContext, flags Flags, deps PathDeps) android.Path { |
| 206 | panic(fmt.Errorf("baseCrater doesn't know how to crate things!")) |
| 207 | } |
| 208 | |
| 209 | func (compiler *baseCompiler) compilerDeps(ctx DepsContext, deps Deps) Deps { |
| 210 | deps.Rlibs = append(deps.Rlibs, compiler.Properties.Rlibs...) |
| 211 | deps.Dylibs = append(deps.Dylibs, compiler.Properties.Dylibs...) |
Matthew Maurer | 0f003b1 | 2020-06-29 14:34:06 -0700 | [diff] [blame] | 212 | deps.Rustlibs = append(deps.Rustlibs, compiler.Properties.Rustlibs...) |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 213 | deps.ProcMacros = append(deps.ProcMacros, compiler.Properties.Proc_macros...) |
| 214 | deps.StaticLibs = append(deps.StaticLibs, compiler.Properties.Static_libs...) |
| 215 | deps.SharedLibs = append(deps.SharedLibs, compiler.Properties.Shared_libs...) |
| 216 | |
Matthew Maurer | 99020b0 | 2019-10-31 10:44:40 -0700 | [diff] [blame] | 217 | if !Bool(compiler.Properties.No_stdlibs) { |
| 218 | for _, stdlib := range config.Stdlibs { |
Jiyong Park | b5d2dd2 | 2020-08-31 17:22:01 +0900 | [diff] [blame^] | 219 | // If we're building for the primary arch of the build host, use the compiler's stdlibs |
| 220 | if ctx.Target().Os == android.BuildOs && ctx.TargetPrimary() { |
Matthew Maurer | 99020b0 | 2019-10-31 10:44:40 -0700 | [diff] [blame] | 221 | stdlib = stdlib + "_" + ctx.toolchain().RustTriple() |
| 222 | } |
| 223 | |
Ivan Lozano | 042504f | 2020-08-18 14:31:23 -0400 | [diff] [blame] | 224 | // For devices, we always link stdlibs in as dylibs except for ffi static libraries. |
| 225 | // (rustc does not support linking libstd as a dylib for ffi static libraries) |
| 226 | if ctx.Host() { |
| 227 | deps.Rustlibs = append(deps.Rustlibs, stdlib) |
| 228 | } else if ctx.RustModule().compiler.static() { |
| 229 | deps.Rlibs = append(deps.Rlibs, stdlib) |
| 230 | } else { |
| 231 | deps.Dylibs = append(deps.Dylibs, stdlib) |
| 232 | } |
Matthew Maurer | 99020b0 | 2019-10-31 10:44:40 -0700 | [diff] [blame] | 233 | } |
| 234 | } |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 235 | return deps |
| 236 | } |
| 237 | |
Ivan Lozano | 45901ed | 2020-07-24 16:05:01 -0400 | [diff] [blame] | 238 | func bionicDeps(deps Deps) Deps { |
Ivan Lozano | f1c8433 | 2019-09-20 11:00:37 -0700 | [diff] [blame] | 239 | deps.SharedLibs = append(deps.SharedLibs, "liblog") |
| 240 | deps.SharedLibs = append(deps.SharedLibs, "libc") |
| 241 | deps.SharedLibs = append(deps.SharedLibs, "libm") |
| 242 | deps.SharedLibs = append(deps.SharedLibs, "libdl") |
| 243 | |
| 244 | //TODO(b/141331117) libstd requires libgcc on Android |
| 245 | deps.StaticLibs = append(deps.StaticLibs, "libgcc") |
| 246 | |
| 247 | return deps |
| 248 | } |
| 249 | |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 250 | func (compiler *baseCompiler) crateName() string { |
| 251 | return compiler.Properties.Crate_name |
| 252 | } |
| 253 | |
Colin Cross | 70dda7e | 2019-10-01 22:05:35 -0700 | [diff] [blame] | 254 | func (compiler *baseCompiler) installDir(ctx ModuleContext) android.InstallPath { |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 255 | dir := compiler.dir |
| 256 | if ctx.toolchain().Is64Bit() && compiler.dir64 != "" { |
| 257 | dir = compiler.dir64 |
| 258 | } |
Ivan Lozano | d6fdca8 | 2020-04-07 12:30:33 -0400 | [diff] [blame] | 259 | if ctx.Target().NativeBridge == android.NativeBridgeEnabled { |
| 260 | dir = filepath.Join(dir, ctx.Target().NativeBridgeRelativePath) |
| 261 | } |
| 262 | if !ctx.Host() && ctx.Config().HasMultilibConflict(ctx.Arch().ArchType) { |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 263 | dir = filepath.Join(dir, ctx.Arch().ArchType.String()) |
| 264 | } |
| 265 | return android.PathForModuleInstall(ctx, dir, compiler.subDir, |
| 266 | compiler.relativeInstallPath(), compiler.relative) |
| 267 | } |
| 268 | |
Ivan Lozano | a0cd8f9 | 2020-04-09 09:56:02 -0400 | [diff] [blame] | 269 | func (compiler *baseCompiler) nativeCoverage() bool { |
| 270 | return false |
| 271 | } |
| 272 | |
Thiébaud Weksteen | fabaff6 | 2020-08-27 13:48:36 +0200 | [diff] [blame] | 273 | func (compiler *baseCompiler) install(ctx ModuleContext) { |
| 274 | path := ctx.RustModule().outputFile |
| 275 | if compiler.strippedOutputFile.Valid() { |
| 276 | path = compiler.strippedOutputFile |
| 277 | } |
| 278 | compiler.path = ctx.InstallFile(compiler.installDir(ctx), path.Path().Base(), path.Path()) |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 279 | } |
| 280 | |
| 281 | func (compiler *baseCompiler) getStem(ctx ModuleContext) string { |
| 282 | return compiler.getStemWithoutSuffix(ctx) + String(compiler.Properties.Suffix) |
| 283 | } |
| 284 | |
| 285 | func (compiler *baseCompiler) getStemWithoutSuffix(ctx BaseModuleContext) string { |
Thiébaud Weksteen | 1f7f70f | 2020-06-24 11:32:48 +0200 | [diff] [blame] | 286 | stem := ctx.ModuleName() |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 287 | if String(compiler.Properties.Stem) != "" { |
| 288 | stem = String(compiler.Properties.Stem) |
| 289 | } |
| 290 | |
| 291 | return stem |
| 292 | } |
Ivan Lozano | ad8b18b | 2019-10-31 19:38:29 -0700 | [diff] [blame] | 293 | |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 294 | func (compiler *baseCompiler) relativeInstallPath() string { |
| 295 | return String(compiler.Properties.Relative_install_path) |
| 296 | } |
| 297 | |
Ivan Lozano | 4384568 | 2020-07-09 21:03:28 -0400 | [diff] [blame] | 298 | // Returns the Path for the main source file along with Paths for generated source files from modules listed in srcs. |
Chih-Hung Hsieh | bbd25ae | 2020-05-15 17:36:30 -0700 | [diff] [blame] | 299 | func srcPathFromModuleSrcs(ctx ModuleContext, srcs []string) (android.Path, android.Paths) { |
| 300 | // The srcs can contain strings with prefix ":". |
| 301 | // They are dependent modules of this module, with android.SourceDepTag. |
| 302 | // They are not the main source file compiled by rustc. |
| 303 | numSrcs := 0 |
| 304 | srcIndex := 0 |
| 305 | for i, s := range srcs { |
| 306 | if android.SrcIsModule(s) == "" { |
| 307 | numSrcs++ |
| 308 | srcIndex = i |
| 309 | } |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 310 | } |
Chih-Hung Hsieh | bbd25ae | 2020-05-15 17:36:30 -0700 | [diff] [blame] | 311 | if numSrcs != 1 { |
Ivan Lozano | 4384568 | 2020-07-09 21:03:28 -0400 | [diff] [blame] | 312 | ctx.PropertyErrorf("srcs", incorrectSourcesError) |
Chih-Hung Hsieh | bbd25ae | 2020-05-15 17:36:30 -0700 | [diff] [blame] | 313 | } |
| 314 | if srcIndex != 0 { |
| 315 | ctx.PropertyErrorf("srcs", "main source file must be the first in srcs") |
| 316 | } |
| 317 | paths := android.PathsForModuleSrc(ctx, srcs) |
Ivan Lozano | 4384568 | 2020-07-09 21:03:28 -0400 | [diff] [blame] | 318 | return paths[srcIndex], paths[1:] |
Ivan Lozano | ffee334 | 2019-08-27 12:03:00 -0700 | [diff] [blame] | 319 | } |