blob: b1f049d15b6b1f4b00a2189ee58336af7a1646e3 [file] [log] [blame]
Ivan Lozanoffee3342019-08-27 12:03:00 -07001// 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
15package rust
16
17import (
Ivan Lozano1776a2a2020-11-11 10:59:52 -050018 "path/filepath"
Ivan Lozanoffee3342019-08-27 12:03:00 -070019 "strings"
20
21 "github.com/google/blueprint"
22
23 "android/soong/android"
Thiébaud Weksteen71512f32020-11-03 15:17:51 +010024 "android/soong/rust/config"
Ivan Lozanoffee3342019-08-27 12:03:00 -070025)
26
27var (
28 _ = pctx.SourcePathVariable("rustcCmd", "${config.RustBin}/rustc")
Peter Collingbournee7c71c32023-03-31 20:21:19 -070029 _ = pctx.SourcePathVariable("mkcraterspCmd", "build/soong/scripts/mkcratersp.py")
Ivan Lozanoffee3342019-08-27 12:03:00 -070030 rustc = pctx.AndroidStaticRule("rustc",
31 blueprint.RuleParams{
Ivan Lozano43845682020-07-09 21:03:28 -040032 Command: "$envVars $rustcCmd " +
Peter Collingbournee7c71c32023-03-31 20:21:19 -070033 "-C linker=$mkcraterspCmd " +
Chih-Hung Hsieh29aa9fd2020-08-13 15:46:21 -070034 "--emit link -o $out --emit dep-info=$out.d.raw $in ${libFlags} $rustcFlags" +
35 " && grep \"^$out:\" $out.d.raw > $out.d",
Peter Collingbournee7c71c32023-03-31 20:21:19 -070036 CommandDeps: []string{"$rustcCmd", "$mkcraterspCmd"},
Ivan Lozanob2df9f82019-11-05 12:16:46 -080037 // Rustc deps-info writes out make compatible dep files: https://github.com/rust-lang/rust/issues/7633
Chih-Hung Hsieh29aa9fd2020-08-13 15:46:21 -070038 // Rustc emits unneeded dependency lines for the .d and input .rs files.
39 // Those extra lines cause ninja warning:
40 // "warning: depfile has multiple output paths"
41 // For ninja, we keep/grep only the dependency rule for the rust $out file.
Ivan Lozanob2df9f82019-11-05 12:16:46 -080042 Deps: blueprint.DepsGCC,
43 Depfile: "$out.d",
Ivan Lozanoffee3342019-08-27 12:03:00 -070044 },
Peter Collingbournee7c71c32023-03-31 20:21:19 -070045 "rustcFlags", "libFlags", "envVars")
46 rustLink = pctx.AndroidStaticRule("rustLink",
47 blueprint.RuleParams{
A. Cody Schuffelenf29ca582023-07-13 19:03:39 -070048 Command: "${config.RustLinker} -o $out ${crtBegin} ${earlyLinkFlags} @$in ${linkFlags} ${crtEnd}",
Peter Collingbournee7c71c32023-03-31 20:21:19 -070049 },
A. Cody Schuffelenf29ca582023-07-13 19:03:39 -070050 "earlyLinkFlags", "linkFlags", "crtBegin", "crtEnd")
Ivan Lozanoa0cd8f92020-04-09 09:56:02 -040051
Dan Albert06feee92021-03-19 15:06:02 -070052 _ = pctx.SourcePathVariable("rustdocCmd", "${config.RustBin}/rustdoc")
53 rustdoc = pctx.AndroidStaticRule("rustdoc",
54 blueprint.RuleParams{
Dan Albertb433bf72021-04-27 17:12:02 -070055 Command: "$envVars $rustdocCmd $rustdocFlags $in -o $outDir && " +
Dan Albert06feee92021-03-19 15:06:02 -070056 "touch $out",
57 CommandDeps: []string{"$rustdocCmd"},
58 },
59 "rustdocFlags", "outDir", "envVars")
60
Thiébaud Weksteen92f703b2020-06-22 13:28:02 +020061 _ = pctx.SourcePathVariable("clippyCmd", "${config.RustBin}/clippy-driver")
62 clippyDriver = pctx.AndroidStaticRule("clippy",
63 blueprint.RuleParams{
Ivan Lozanobae62be2020-07-21 13:28:27 -040064 Command: "$envVars $clippyCmd " +
Thiébaud Weksteen92f703b2020-06-22 13:28:02 +020065 // Because clippy-driver uses rustc as backend, we need to have some output even during the linting.
66 // Use the metadata output as it has the smallest footprint.
Thiébaud Weksteen94c83252021-04-07 16:00:19 +020067 "--emit metadata -o $out --emit dep-info=$out.d.raw $in ${libFlags} " +
68 "$rustcFlags $clippyFlags" +
69 " && grep \"^$out:\" $out.d.raw > $out.d",
Thiébaud Weksteen92f703b2020-06-22 13:28:02 +020070 CommandDeps: []string{"$clippyCmd"},
Thiébaud Weksteen94c83252021-04-07 16:00:19 +020071 Deps: blueprint.DepsGCC,
72 Depfile: "$out.d",
Thiébaud Weksteen92f703b2020-06-22 13:28:02 +020073 },
Ivan Lozanobae62be2020-07-21 13:28:27 -040074 "rustcFlags", "libFlags", "clippyFlags", "envVars")
Thiébaud Weksteen92f703b2020-06-22 13:28:02 +020075
Ivan Lozanoa0cd8f92020-04-09 09:56:02 -040076 zip = pctx.AndroidStaticRule("zip",
77 blueprint.RuleParams{
78 Command: "cat $out.rsp | tr ' ' '\\n' | tr -d \\' | sort -u > ${out}.tmp && ${SoongZipCmd} -o ${out} -C $$OUT_DIR -l ${out}.tmp",
79 CommandDeps: []string{"${SoongZipCmd}"},
80 Rspfile: "$out.rsp",
81 RspfileContent: "$in",
82 })
Ivan Lozano43845682020-07-09 21:03:28 -040083
84 cp = pctx.AndroidStaticRule("cp",
85 blueprint.RuleParams{
86 Command: "cp `cat $outDir.rsp` $outDir",
87 Rspfile: "${outDir}.rsp",
88 RspfileContent: "$in",
89 },
90 "outDir")
Sasha Smundaka76acba2022-04-18 20:12:56 -070091
92 // Cross-referencing:
93 _ = pctx.SourcePathVariable("rustExtractor",
94 "prebuilts/build-tools/${config.HostPrebuiltTag}/bin/rust_extractor")
95 _ = pctx.VariableFunc("kytheCorpus",
96 func(ctx android.PackageVarContext) string { return ctx.Config().XrefCorpusName() })
97 _ = pctx.VariableFunc("kytheCuEncoding",
98 func(ctx android.PackageVarContext) string { return ctx.Config().XrefCuEncoding() })
99 _ = pctx.SourcePathVariable("kytheVnames", "build/soong/vnames.json")
100 kytheExtract = pctx.AndroidStaticRule("kythe",
101 blueprint.RuleParams{
102 Command: `KYTHE_CORPUS=${kytheCorpus} ` +
103 `KYTHE_OUTPUT_FILE=$out ` +
104 `KYTHE_VNAMES=$kytheVnames ` +
105 `KYTHE_KZIP_ENCODING=${kytheCuEncoding} ` +
106 `KYTHE_CANONICALIZE_VNAME_PATHS=prefer-relative ` +
107 `$rustExtractor $envVars ` +
108 `$rustcCmd ` +
Peter Collingbournee7c71c32023-03-31 20:21:19 -0700109 `-C linker=true ` +
Sasha Smundaka76acba2022-04-18 20:12:56 -0700110 `$in ${libFlags} $rustcFlags`,
111 CommandDeps: []string{"$rustExtractor", "$kytheVnames"},
112 Rspfile: "${out}.rsp",
113 RspfileContent: "$in",
114 },
Peter Collingbournee7c71c32023-03-31 20:21:19 -0700115 "rustcFlags", "libFlags", "envVars")
Ivan Lozanoffee3342019-08-27 12:03:00 -0700116)
117
Ivan Lozanoa0cd8f92020-04-09 09:56:02 -0400118type buildOutput struct {
Joel Galensonfa049382021-01-14 16:03:18 -0800119 outputFile android.Path
Sasha Smundaka76acba2022-04-18 20:12:56 -0700120 kytheFile android.Path
Ivan Lozanoa0cd8f92020-04-09 09:56:02 -0400121}
Ivan Lozanoffee3342019-08-27 12:03:00 -0700122
Ivan Lozanoa0cd8f92020-04-09 09:56:02 -0400123func init() {
124 pctx.HostBinToolVariable("SoongZipCmd", "soong_zip")
Ivan Lozanoffee3342019-08-27 12:03:00 -0700125}
126
Thiébaud Weksteen1f7f70f2020-06-24 11:32:48 +0200127func TransformSrcToBinary(ctx ModuleContext, mainSrc android.Path, deps PathDeps, flags Flags,
Dan Albert06feee92021-03-19 15:06:02 -0700128 outputFile android.WritablePath) buildOutput {
Pirama Arumuga Nainarf77913f2021-10-25 15:37:43 -0700129 flags.GlobalRustFlags = append(flags.GlobalRustFlags, "-C lto=thin")
Ivan Lozano31b095d2019-11-20 10:14:33 -0800130
Dan Albert06feee92021-03-19 15:06:02 -0700131 return transformSrctoCrate(ctx, mainSrc, deps, flags, outputFile, "bin")
Ivan Lozanoffee3342019-08-27 12:03:00 -0700132}
133
Thiébaud Weksteen1f7f70f2020-06-24 11:32:48 +0200134func TransformSrctoRlib(ctx ModuleContext, mainSrc android.Path, deps PathDeps, flags Flags,
Dan Albert06feee92021-03-19 15:06:02 -0700135 outputFile android.WritablePath) buildOutput {
136 return transformSrctoCrate(ctx, mainSrc, deps, flags, outputFile, "rlib")
Ivan Lozanoffee3342019-08-27 12:03:00 -0700137}
138
Thiébaud Weksteen1f7f70f2020-06-24 11:32:48 +0200139func TransformSrctoDylib(ctx ModuleContext, mainSrc android.Path, deps PathDeps, flags Flags,
Dan Albert06feee92021-03-19 15:06:02 -0700140 outputFile android.WritablePath) buildOutput {
Chris Wailes5f788402023-03-02 16:06:01 -0800141 flags.GlobalRustFlags = append(flags.GlobalRustFlags, "-C lto=thin")
142
Dan Albert06feee92021-03-19 15:06:02 -0700143 return transformSrctoCrate(ctx, mainSrc, deps, flags, outputFile, "dylib")
Ivan Lozanoffee3342019-08-27 12:03:00 -0700144}
145
Thiébaud Weksteen1f7f70f2020-06-24 11:32:48 +0200146func TransformSrctoStatic(ctx ModuleContext, mainSrc android.Path, deps PathDeps, flags Flags,
Dan Albert06feee92021-03-19 15:06:02 -0700147 outputFile android.WritablePath) buildOutput {
Pirama Arumuga Nainarf77913f2021-10-25 15:37:43 -0700148 flags.GlobalRustFlags = append(flags.GlobalRustFlags, "-C lto=thin")
Dan Albert06feee92021-03-19 15:06:02 -0700149 return transformSrctoCrate(ctx, mainSrc, deps, flags, outputFile, "staticlib")
Ivan Lozano52767be2019-10-18 14:49:46 -0700150}
151
Thiébaud Weksteen1f7f70f2020-06-24 11:32:48 +0200152func TransformSrctoShared(ctx ModuleContext, mainSrc android.Path, deps PathDeps, flags Flags,
Dan Albert06feee92021-03-19 15:06:02 -0700153 outputFile android.WritablePath) buildOutput {
Pirama Arumuga Nainarf77913f2021-10-25 15:37:43 -0700154 flags.GlobalRustFlags = append(flags.GlobalRustFlags, "-C lto=thin")
Dan Albert06feee92021-03-19 15:06:02 -0700155 return transformSrctoCrate(ctx, mainSrc, deps, flags, outputFile, "cdylib")
Ivan Lozano52767be2019-10-18 14:49:46 -0700156}
157
Thiébaud Weksteen1f7f70f2020-06-24 11:32:48 +0200158func TransformSrctoProcMacro(ctx ModuleContext, mainSrc android.Path, deps PathDeps,
Dan Albert06feee92021-03-19 15:06:02 -0700159 flags Flags, outputFile android.WritablePath) buildOutput {
160 return transformSrctoCrate(ctx, mainSrc, deps, flags, outputFile, "proc-macro")
Ivan Lozanoffee3342019-08-27 12:03:00 -0700161}
162
163func rustLibsToPaths(libs RustLibraries) android.Paths {
164 var paths android.Paths
165 for _, lib := range libs {
166 paths = append(paths, lib.Path)
167 }
168 return paths
169}
170
Dan Albert06feee92021-03-19 15:06:02 -0700171func makeLibFlags(deps PathDeps) []string {
172 var libFlags []string
173
174 // Collect library/crate flags
175 for _, lib := range deps.RLibs {
176 libFlags = append(libFlags, "--extern "+lib.CrateName+"="+lib.Path.String())
177 }
178 for _, lib := range deps.DyLibs {
179 libFlags = append(libFlags, "--extern "+lib.CrateName+"="+lib.Path.String())
180 }
181 for _, proc_macro := range deps.ProcMacros {
182 libFlags = append(libFlags, "--extern "+proc_macro.CrateName+"="+proc_macro.Path.String())
183 }
184
185 for _, path := range deps.linkDirs {
186 libFlags = append(libFlags, "-L "+path)
187 }
188
189 return libFlags
190}
191
192func rustEnvVars(ctx ModuleContext, deps PathDeps) []string {
193 var envVars []string
194
195 // libstd requires a specific environment variable to be set. This is
196 // not officially documented and may be removed in the future. See
197 // https://github.com/rust-lang/rust/blob/master/library/std/src/env.rs#L866.
198 if ctx.RustModule().CrateName() == "std" {
199 envVars = append(envVars, "STD_ENV_ARCH="+config.StdEnvArch[ctx.RustModule().Arch().ArchType])
200 }
201
202 if len(deps.SrcDeps) > 0 {
203 moduleGenDir := ctx.RustModule().compiler.CargoOutDir()
204 // We must calculate an absolute path for OUT_DIR since Rust's include! macro (which normally consumes this)
205 // assumes that paths are relative to the source file.
206 var outDirPrefix string
207 if !filepath.IsAbs(moduleGenDir.String()) {
208 // If OUT_DIR is not absolute, we use $$PWD to generate an absolute path (os.Getwd() returns '/')
209 outDirPrefix = "$$PWD/"
210 } else {
211 // If OUT_DIR is absolute, then moduleGenDir will be an absolute path, so we don't need to set this to anything.
212 outDirPrefix = ""
213 }
214 envVars = append(envVars, "OUT_DIR="+filepath.Join(outDirPrefix, moduleGenDir.String()))
Peter Collingbourne0dcd62e2023-03-31 23:05:16 -0700215 } else {
216 // TODO(pcc): Change this to "OUT_DIR=" after fixing crates to not rely on this value.
217 envVars = append(envVars, "OUT_DIR=out")
Dan Albert06feee92021-03-19 15:06:02 -0700218 }
219
Matthew Maurer34609fa2023-06-26 21:10:13 +0000220 envVars = append(envVars, "ANDROID_RUST_VERSION="+config.GetRustVersion(ctx))
221
222 if ctx.RustModule().compiler.CargoEnvCompat() {
223 if bin, ok := ctx.RustModule().compiler.(*binaryDecorator); ok {
224 envVars = append(envVars, "CARGO_BIN_NAME="+bin.getStem(ctx))
225 }
226 envVars = append(envVars, "CARGO_CRATE_NAME="+ctx.RustModule().CrateName())
227 envVars = append(envVars, "CARGO_PKG_NAME="+ctx.RustModule().CrateName())
228 pkgVersion := ctx.RustModule().compiler.CargoPkgVersion()
229 if pkgVersion != "" {
230 envVars = append(envVars, "CARGO_PKG_VERSION="+pkgVersion)
Ivan Lozanof4455622023-07-28 12:42:20 -0400231
232 // Ensure the version is in the form of "x.y.z" (approximately semver compliant).
233 //
234 // For our purposes, we don't care to enforce that these are integers since they may
235 // include other characters at times (e.g. sometimes the patch version is more than an integer).
236 if strings.Count(pkgVersion, ".") == 2 {
237 var semver_parts = strings.Split(pkgVersion, ".")
238 envVars = append(envVars, "CARGO_PKG_VERSION_MAJOR="+semver_parts[0])
239 envVars = append(envVars, "CARGO_PKG_VERSION_MINOR="+semver_parts[1])
240 envVars = append(envVars, "CARGO_PKG_VERSION_PATCH="+semver_parts[2])
241 }
Matthew Maurer34609fa2023-06-26 21:10:13 +0000242 }
243 }
244
245 envVars = append(envVars, "AR=${cc_config.ClangBin}/llvm-ar")
246
A. Cody Schuffelenf29ca582023-07-13 19:03:39 -0700247 if ctx.Darwin() {
248 envVars = append(envVars, "ANDROID_RUST_DARWIN=true")
249 }
250
Dan Albert06feee92021-03-19 15:06:02 -0700251 return envVars
252}
253
Thiébaud Weksteen1f7f70f2020-06-24 11:32:48 +0200254func transformSrctoCrate(ctx ModuleContext, main android.Path, deps PathDeps, flags Flags,
Ivan Lozano7b0781d2021-11-03 15:30:18 -0400255 outputFile android.WritablePath, crateType string) buildOutput {
Ivan Lozanoffee3342019-08-27 12:03:00 -0700256
257 var inputs android.Paths
Peter Collingbournee7c71c32023-03-31 20:21:19 -0700258 var implicits, linkImplicits, linkOrderOnly android.Paths
Ivan Lozanoa0cd8f92020-04-09 09:56:02 -0400259 var output buildOutput
Dan Albert06feee92021-03-19 15:06:02 -0700260 var rustcFlags, linkFlags []string
A. Cody Schuffelenf29ca582023-07-13 19:03:39 -0700261 var earlyLinkFlags string
Ivan Lozanoa0cd8f92020-04-09 09:56:02 -0400262
263 output.outputFile = outputFile
Jaewoong Jung18aefc12020-12-21 09:11:10 -0800264 crateName := ctx.RustModule().CrateName()
Thiébaud Weksteen1f7f70f2020-06-24 11:32:48 +0200265 targetTriple := ctx.toolchain().RustTriple()
Ivan Lozanoffee3342019-08-27 12:03:00 -0700266
Dan Albert06feee92021-03-19 15:06:02 -0700267 envVars := rustEnvVars(ctx, deps)
Thiébaud Weksteen71512f32020-11-03 15:17:51 +0100268
Ivan Lozanoffee3342019-08-27 12:03:00 -0700269 inputs = append(inputs, main)
270
271 // Collect rustc flags
Ivan Lozanof1c84332019-09-20 11:00:37 -0700272 rustcFlags = append(rustcFlags, flags.GlobalRustFlags...)
Ivan Lozanoffee3342019-08-27 12:03:00 -0700273 rustcFlags = append(rustcFlags, flags.RustFlags...)
Ivan Lozano7b0781d2021-11-03 15:30:18 -0400274 rustcFlags = append(rustcFlags, "--crate-type="+crateType)
Jaewoong Jung18aefc12020-12-21 09:11:10 -0800275 if crateName != "" {
276 rustcFlags = append(rustcFlags, "--crate-name="+crateName)
Ivan Lozanoad8b18b2019-10-31 19:38:29 -0700277 }
Ivan Lozanoffee3342019-08-27 12:03:00 -0700278 if targetTriple != "" {
279 rustcFlags = append(rustcFlags, "--target="+targetTriple)
Ivan Lozanof1c84332019-09-20 11:00:37 -0700280 linkFlags = append(linkFlags, "-target "+targetTriple)
Ivan Lozanoffee3342019-08-27 12:03:00 -0700281 }
Matthew Maurerbb3add12020-06-25 09:34:12 -0700282
283 // Suppress an implicit sysroot
284 rustcFlags = append(rustcFlags, "--sysroot=/dev/null")
285
Chris Wailesd9781fd2021-12-03 17:17:28 -0800286 // Enable incremental compilation if requested by user
287 if ctx.Config().IsEnvTrue("SOONG_RUSTC_INCREMENTAL") {
288 incrementalPath := android.PathForOutput(ctx, "rustc").String()
289
Chris Wailes6d12db42023-02-24 16:58:18 -0800290 rustcFlags = append(rustcFlags, "-Cincremental="+incrementalPath)
291 }
292
293 // Disallow experimental features
294 modulePath := android.PathForModuleSrc(ctx).String()
295 if !(android.IsThirdPartyPath(modulePath) || strings.HasPrefix(modulePath, "prebuilts")) {
Chris Wailes547bfdd2023-05-31 11:53:44 -0700296 rustcFlags = append(rustcFlags, "-Zallow-features=\"\"")
Chris Wailesd9781fd2021-12-03 17:17:28 -0800297 }
298
Ivan Lozanof1c84332019-09-20 11:00:37 -0700299 // Collect linker flags
A. Cody Schuffelenf29ca582023-07-13 19:03:39 -0700300 if !ctx.Darwin() {
301 earlyLinkFlags = "-Wl,--as-needed"
302 }
303
Ivan Lozanof1c84332019-09-20 11:00:37 -0700304 linkFlags = append(linkFlags, flags.GlobalLinkFlags...)
305 linkFlags = append(linkFlags, flags.LinkFlags...)
Ivan Lozanoffee3342019-08-27 12:03:00 -0700306
Ivan Lozanoa2268632021-07-22 10:52:06 -0400307 // Check if this module needs to use the bootstrap linker
308 if ctx.RustModule().Bootstrap() && !ctx.RustModule().InRecovery() && !ctx.RustModule().InRamdisk() && !ctx.RustModule().InVendorRamdisk() {
309 dynamicLinker := "-Wl,-dynamic-linker,/system/bin/bootstrap/linker"
310 if ctx.toolchain().Is64Bit() {
311 dynamicLinker += "64"
312 }
313 linkFlags = append(linkFlags, dynamicLinker)
314 }
315
Dan Albert06feee92021-03-19 15:06:02 -0700316 libFlags := makeLibFlags(deps)
Ivan Lozanoffee3342019-08-27 12:03:00 -0700317
318 // Collect dependencies
Ivan Lozanob2df9f82019-11-05 12:16:46 -0800319 implicits = append(implicits, rustLibsToPaths(deps.RLibs)...)
320 implicits = append(implicits, rustLibsToPaths(deps.DyLibs)...)
321 implicits = append(implicits, rustLibsToPaths(deps.ProcMacros)...)
Yi Kong46c6e592022-01-20 22:55:00 +0800322 implicits = append(implicits, deps.AfdoProfiles...)
Peter Collingbournee7c71c32023-03-31 20:21:19 -0700323 implicits = append(implicits, deps.srcProviderFiles...)
324 implicits = append(implicits, deps.WholeStaticLibs...)
Ivan Lozano43845682020-07-09 21:03:28 -0400325
Peter Collingbournee7c71c32023-03-31 20:21:19 -0700326 linkImplicits = append(linkImplicits, deps.LibDeps...)
327 linkImplicits = append(linkImplicits, deps.CrtBegin...)
328 linkImplicits = append(linkImplicits, deps.CrtEnd...)
Ivan Lozanoffee3342019-08-27 12:03:00 -0700329
Peter Collingbournee7c71c32023-03-31 20:21:19 -0700330 linkOrderOnly = append(linkOrderOnly, deps.linkObjects...)
Sam Delmerico51d6d1c2023-03-28 16:54:00 -0400331
Ivan Lozano43845682020-07-09 21:03:28 -0400332 if len(deps.SrcDeps) > 0 {
Thiébaud Weksteenee6a89b2021-02-25 16:30:57 +0100333 moduleGenDir := ctx.RustModule().compiler.CargoOutDir()
Ivan Lozano43845682020-07-09 21:03:28 -0400334 var outputs android.WritablePaths
335
336 for _, genSrc := range deps.SrcDeps {
Ivan Lozano10735d92020-07-22 09:14:47 -0400337 if android.SuffixInList(outputs.Strings(), genSubDir+genSrc.Base()) {
Ivan Lozano43845682020-07-09 21:03:28 -0400338 ctx.PropertyErrorf("srcs",
339 "multiple source providers generate the same filename output: "+genSrc.Base())
340 }
Ivan Lozano10735d92020-07-22 09:14:47 -0400341 outputs = append(outputs, android.PathForModuleOut(ctx, genSubDir+genSrc.Base()))
Ivan Lozano43845682020-07-09 21:03:28 -0400342 }
343
344 ctx.Build(pctx, android.BuildParams{
345 Rule: cp,
Thiébaud Weksteenee6a89b2021-02-25 16:30:57 +0100346 Description: "cp " + moduleGenDir.Path().Rel(),
Ivan Lozano43845682020-07-09 21:03:28 -0400347 Outputs: outputs,
348 Inputs: deps.SrcDeps,
349 Args: map[string]string{
350 "outDir": moduleGenDir.String(),
351 },
352 })
353 implicits = append(implicits, outputs.Paths()...)
Ivan Lozano43845682020-07-09 21:03:28 -0400354 }
355
Ivan Lozanobae62be2020-07-21 13:28:27 -0400356 if flags.Clippy {
357 clippyFile := android.PathForModuleOut(ctx, outputFile.Base()+".clippy")
358 ctx.Build(pctx, android.BuildParams{
Peter Collingbournee7c71c32023-03-31 20:21:19 -0700359 Rule: clippyDriver,
360 Description: "clippy " + main.Rel(),
361 Output: clippyFile,
362 Inputs: inputs,
363 Implicits: implicits,
Ivan Lozanobae62be2020-07-21 13:28:27 -0400364 Args: map[string]string{
365 "rustcFlags": strings.Join(rustcFlags, " "),
366 "libFlags": strings.Join(libFlags, " "),
367 "clippyFlags": strings.Join(flags.ClippyFlags, " "),
368 "envVars": strings.Join(envVars, " "),
369 },
370 })
371 // Declare the clippy build as an implicit dependency of the original crate.
372 implicits = append(implicits, clippyFile)
373 }
374
Peter Collingbournee7c71c32023-03-31 20:21:19 -0700375 rustcOutputFile := outputFile
376 usesLinker := crateType == "bin" || crateType == "dylib" || crateType == "cdylib" || crateType == "proc-macro"
377 if usesLinker {
378 rustcOutputFile = android.PathForModuleOut(ctx, outputFile.Base()+".rsp")
379 }
380
Ivan Lozanoffee3342019-08-27 12:03:00 -0700381 ctx.Build(pctx, android.BuildParams{
Peter Collingbournee7c71c32023-03-31 20:21:19 -0700382 Rule: rustc,
383 Description: "rustc " + main.Rel(),
384 Output: rustcOutputFile,
385 Inputs: inputs,
386 Implicits: implicits,
Ivan Lozanoffee3342019-08-27 12:03:00 -0700387 Args: map[string]string{
388 "rustcFlags": strings.Join(rustcFlags, " "),
Ivan Lozanoffee3342019-08-27 12:03:00 -0700389 "libFlags": strings.Join(libFlags, " "),
Ivan Lozano43845682020-07-09 21:03:28 -0400390 "envVars": strings.Join(envVars, " "),
Ivan Lozanoffee3342019-08-27 12:03:00 -0700391 },
392 })
393
Peter Collingbournee7c71c32023-03-31 20:21:19 -0700394 if usesLinker {
395 ctx.Build(pctx, android.BuildParams{
396 Rule: rustLink,
397 Description: "rustLink " + main.Rel(),
398 Output: outputFile,
399 Inputs: android.Paths{rustcOutputFile},
400 Implicits: linkImplicits,
401 OrderOnly: linkOrderOnly,
402 Args: map[string]string{
A. Cody Schuffelenf29ca582023-07-13 19:03:39 -0700403 "earlyLinkFlags": earlyLinkFlags,
404 "linkFlags": strings.Join(linkFlags, " "),
405 "crtBegin": strings.Join(deps.CrtBegin.Strings(), " "),
406 "crtEnd": strings.Join(deps.CrtEnd.Strings(), " "),
Peter Collingbournee7c71c32023-03-31 20:21:19 -0700407 },
408 })
409 }
410
Sasha Smundaka76acba2022-04-18 20:12:56 -0700411 if flags.EmitXrefs {
412 kytheFile := android.PathForModuleOut(ctx, outputFile.Base()+".kzip")
413 ctx.Build(pctx, android.BuildParams{
414 Rule: kytheExtract,
415 Description: "Xref Rust extractor " + main.Rel(),
416 Output: kytheFile,
417 Inputs: inputs,
418 Implicits: implicits,
419 Args: map[string]string{
420 "rustcFlags": strings.Join(rustcFlags, " "),
Sasha Smundaka76acba2022-04-18 20:12:56 -0700421 "libFlags": strings.Join(libFlags, " "),
Sasha Smundaka76acba2022-04-18 20:12:56 -0700422 "envVars": strings.Join(envVars, " "),
423 },
424 })
425 output.kytheFile = kytheFile
426 }
Ivan Lozanoa0cd8f92020-04-09 09:56:02 -0400427 return output
428}
Dan Albert06feee92021-03-19 15:06:02 -0700429
430func Rustdoc(ctx ModuleContext, main android.Path, deps PathDeps,
431 flags Flags) android.ModuleOutPath {
432
433 rustdocFlags := append([]string{}, flags.RustdocFlags...)
434 rustdocFlags = append(rustdocFlags, "--sysroot=/dev/null")
435
Dan Albertb433bf72021-04-27 17:12:02 -0700436 // Build an index for all our crates. -Z unstable options is required to use
437 // this flag.
438 rustdocFlags = append(rustdocFlags, "-Z", "unstable-options", "--enable-index-page")
439
Dan Albert06feee92021-03-19 15:06:02 -0700440 targetTriple := ctx.toolchain().RustTriple()
441
442 // Collect rustc flags
443 if targetTriple != "" {
444 rustdocFlags = append(rustdocFlags, "--target="+targetTriple)
445 }
446
447 crateName := ctx.RustModule().CrateName()
Dan Albertb433bf72021-04-27 17:12:02 -0700448 rustdocFlags = append(rustdocFlags, "--crate-name "+crateName)
Dan Albert06feee92021-03-19 15:06:02 -0700449
450 rustdocFlags = append(rustdocFlags, makeLibFlags(deps)...)
451 docTimestampFile := android.PathForModuleOut(ctx, "rustdoc.timestamp")
Dan Albertb433bf72021-04-27 17:12:02 -0700452
Chris Wailesb2703ad2021-07-30 13:25:42 -0700453 // Silence warnings about renamed lints for third-party crates
454 modulePath := android.PathForModuleSrc(ctx).String()
455 if android.IsThirdPartyPath(modulePath) {
Chris Wailes7b3eb242023-02-14 16:09:49 -0800456 rustdocFlags = append(rustdocFlags, " -A warnings")
Chris Wailesb2703ad2021-07-30 13:25:42 -0700457 }
Chris Wailes9953a192021-07-28 12:07:16 -0700458
Dan Albertb433bf72021-04-27 17:12:02 -0700459 // Yes, the same out directory is used simultaneously by all rustdoc builds.
460 // This is what cargo does. The docs for individual crates get generated to
461 // a subdirectory named for the crate, and rustdoc synchronizes writes to
462 // shared pieces like the index and search data itself.
463 // https://github.com/rust-lang/rust/blob/master/src/librustdoc/html/render/write_shared.rs#L144-L146
464 docDir := android.PathForOutput(ctx, "rustdoc")
Dan Albert06feee92021-03-19 15:06:02 -0700465
466 ctx.Build(pctx, android.BuildParams{
467 Rule: rustdoc,
468 Description: "rustdoc " + main.Rel(),
469 Output: docTimestampFile,
470 Input: main,
Ivan Lozano8d10fc32021-11-05 16:36:47 -0400471 Implicit: ctx.RustModule().UnstrippedOutputFile(),
Dan Albert06feee92021-03-19 15:06:02 -0700472 Args: map[string]string{
473 "rustdocFlags": strings.Join(rustdocFlags, " "),
474 "outDir": docDir.String(),
475 "envVars": strings.Join(rustEnvVars(ctx, deps), " "),
476 },
477 })
478
479 return docTimestampFile
480}