1 //! A build dependency for Cargo libraries to find system artifacts through the
2 //! `pkg-config` utility.
3 //!
4 //! This library will shell out to `pkg-config` as part of build scripts and
5 //! probe the system to determine how to link to a specified library. The
6 //! `Config` structure serves as a method of configuring how `pkg-config` is
7 //! invoked in a builder style.
8 //!
9 //! A number of environment variables are available to globally configure how
10 //! this crate will invoke `pkg-config`:
11 //!
12 //! * `FOO_NO_PKG_CONFIG` - if set, this will disable running `pkg-config` when
13 //! probing for the library named `foo`.
14 //!
15 //! * `PKG_CONFIG_ALLOW_CROSS` - The `pkg-config` command usually doesn't
16 //! support cross-compilation, and this crate prevents it from selecting
17 //! incompatible versions of libraries.
18 //! Setting `PKG_CONFIG_ALLOW_CROSS=1` disables this protection, which is
19 //! likely to cause linking errors, unless `pkg-config` has been configured
20 //! to use appropriate sysroot and search paths for the target platform.
21 //!
22 //! There are also a number of environment variables which can configure how a
23 //! library is linked to (dynamically vs statically). These variables control
24 //! whether the `--static` flag is passed. Note that this behavior can be
25 //! overridden by configuring explicitly on `Config`. The variables are checked
26 //! in the following order:
27 //!
28 //! * `FOO_STATIC` - pass `--static` for the library `foo`
29 //! * `FOO_DYNAMIC` - do not pass `--static` for the library `foo`
30 //! * `PKG_CONFIG_ALL_STATIC` - pass `--static` for all libraries
31 //! * `PKG_CONFIG_ALL_DYNAMIC` - do not pass `--static` for all libraries
32 //!
33 //! After running `pkg-config` all appropriate Cargo metadata will be printed on
34 //! stdout if the search was successful.
35 //!
36 //! # Example
37 //!
38 //! Find the system library named `foo`, with minimum version 1.2.3:
39 //!
40 //! ```no_run
41 //! fn main() {
42 //! pkg_config::Config::new().atleast_version("1.2.3").probe("foo").unwrap();
43 //! }
44 //! ```
45 //!
46 //! Find the system library named `foo`, with no version requirement (not
47 //! recommended):
48 //!
49 //! ```no_run
50 //! fn main() {
51 //! pkg_config::probe_library("foo").unwrap();
52 //! }
53 //! ```
54 //!
55 //! Configure how library `foo` is linked to.
56 //!
57 //! ```no_run
58 //! fn main() {
59 //! pkg_config::Config::new().atleast_version("1.2.3").statik(true).probe("foo").unwrap();
60 //! }
61 //! ```
62
63 #![doc(html_root_url = "https://docs.rs/pkg-config/0.3")]
64
65 use std::collections::HashMap;
66 use std::env;
67 use std::error;
68 use std::ffi::{OsStr, OsString};
69 use std::fmt;
70 use std::io;
71 use std::ops::{Bound, RangeBounds};
72 use std::path::PathBuf;
73 use std::process::{Command, Output};
74 use std::str;
75
76 #[derive(Clone, Debug)]
77 pub struct Config {
78 statik: Option<bool>,
79 min_version: Bound<String>,
80 max_version: Bound<String>,
81 extra_args: Vec<OsString>,
82 cargo_metadata: bool,
83 env_metadata: bool,
84 print_system_libs: bool,
85 print_system_cflags: bool,
86 }
87
88 #[derive(Clone, Debug)]
89 pub struct Library {
90 /// Libraries specified by -l
91 pub libs: Vec<String>,
92 /// Library search paths specified by -L
93 pub link_paths: Vec<PathBuf>,
94 /// Library file paths specified without -l
95 pub link_files: Vec<PathBuf>,
96 /// Darwin frameworks specified by -framework
97 pub frameworks: Vec<String>,
98 /// Darwin framework search paths specified by -F
99 pub framework_paths: Vec<PathBuf>,
100 /// C/C++ header include paths specified by -I
101 pub include_paths: Vec<PathBuf>,
102 /// Linker options specified by -Wl
103 pub ld_args: Vec<Vec<String>>,
104 /// C/C++ definitions specified by -D
105 pub defines: HashMap<String, Option<String>>,
106 /// Version specified by .pc file's Version field
107 pub version: String,
108 /// Ensure that this struct can only be created via its private `[Library::new]` constructor.
109 /// Users of this crate can only access the struct via `[Config::probe]`.
110 _priv: (),
111 }
112
113 /// Represents all reasons `pkg-config` might not succeed or be run at all.
114 pub enum Error {
115 /// Aborted because of `*_NO_PKG_CONFIG` environment variable.
116 ///
117 /// Contains the name of the responsible environment variable.
118 EnvNoPkgConfig(String),
119
120 /// Detected cross compilation without a custom sysroot.
121 ///
122 /// Ignore the error with `PKG_CONFIG_ALLOW_CROSS=1`,
123 /// which may let `pkg-config` select libraries
124 /// for the host's architecture instead of the target's.
125 CrossCompilation,
126
127 /// Failed to run `pkg-config`.
128 ///
129 /// Contains the command and the cause.
130 Command { command: String, cause: io::Error },
131
132 /// `pkg-config` did not exit sucessfully after probing a library.
133 ///
134 /// Contains the command and output.
135 Failure { command: String, output: Output },
136
137 /// `pkg-config` did not exit sucessfully on the first attempt to probe a library.
138 ///
139 /// Contains the command and output.
140 ProbeFailure {
141 name: String,
142 command: String,
143 output: Output,
144 },
145
146 #[doc(hidden)]
147 // please don't match on this, we're likely to add more variants over time
148 __Nonexhaustive,
149 }
150
151 impl error::Error for Error {}
152
153 impl fmt::Debug for Error {
fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error>154 fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
155 // Failed `unwrap()` prints Debug representation, but the default debug format lacks helpful instructions for the end users
156 <Error as fmt::Display>::fmt(self, f)
157 }
158 }
159
160 impl fmt::Display for Error {
fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error>161 fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
162 match *self {
163 Error::EnvNoPkgConfig(ref name) => write!(f, "Aborted because {} is set", name),
164 Error::CrossCompilation => f.write_str(
165 "pkg-config has not been configured to support cross-compilation.\n\
166 \n\
167 Install a sysroot for the target platform and configure it via\n\
168 PKG_CONFIG_SYSROOT_DIR and PKG_CONFIG_PATH, or install a\n\
169 cross-compiling wrapper for pkg-config and set it via\n\
170 PKG_CONFIG environment variable.",
171 ),
172 Error::Command {
173 ref command,
174 ref cause,
175 } => {
176 match cause.kind() {
177 io::ErrorKind::NotFound => {
178 let crate_name =
179 std::env::var("CARGO_PKG_NAME").unwrap_or_else(|_| "sys".to_owned());
180 let instructions = if cfg!(target_os = "macos") || cfg!(target_os = "ios") {
181 "Try `brew install pkg-config` if you have Homebrew.\n"
182 } else if cfg!(unix) {
183 "Try `apt install pkg-config`, or `yum install pkg-config`,\n\
184 or `pkg install pkg-config`, or `apk add pkgconfig` \
185 depending on your distribution.\n"
186 } else {
187 "" // There's no easy fix for Windows users
188 };
189 write!(f, "Could not run `{command}`\n\
190 The pkg-config command could not be found.\n\
191 \n\
192 Most likely, you need to install a pkg-config package for your OS.\n\
193 {instructions}\
194 \n\
195 If you've already installed it, ensure the pkg-config command is one of the\n\
196 directories in the PATH environment variable.\n\
197 \n\
198 If you did not expect this build to link to a pre-installed system library,\n\
199 then check documentation of the {crate_name} crate for an option to\n\
200 build the library from source, or disable features or dependencies\n\
201 that require pkg-config.", command = command, instructions = instructions, crate_name = crate_name)
202 }
203 _ => write!(f, "Failed to run command `{}`, because: {}", command, cause),
204 }
205 }
206 Error::ProbeFailure {
207 ref name,
208 ref command,
209 ref output,
210 } => {
211 write!(
212 f,
213 "`{}` did not exit successfully: {}\nerror: could not find system library '{}' required by the '{}' crate\n",
214 command, output.status, name, env::var("CARGO_PKG_NAME").unwrap_or_default(),
215 )?;
216 format_output(output, f)
217 }
218 Error::Failure {
219 ref command,
220 ref output,
221 } => {
222 write!(
223 f,
224 "`{}` did not exit successfully: {}",
225 command, output.status
226 )?;
227 format_output(output, f)
228 }
229 Error::__Nonexhaustive => panic!(),
230 }
231 }
232 }
233
format_output(output: &Output, f: &mut fmt::Formatter) -> fmt::Result234 fn format_output(output: &Output, f: &mut fmt::Formatter) -> fmt::Result {
235 let stdout = String::from_utf8_lossy(&output.stdout);
236 if !stdout.is_empty() {
237 write!(f, "\n--- stdout\n{}", stdout)?;
238 }
239 let stderr = String::from_utf8_lossy(&output.stderr);
240 if !stderr.is_empty() {
241 write!(f, "\n--- stderr\n{}", stderr)?;
242 }
243 Ok(())
244 }
245
246 /// Deprecated in favor of the probe_library function
247 #[doc(hidden)]
find_library(name: &str) -> Result<Library, String>248 pub fn find_library(name: &str) -> Result<Library, String> {
249 probe_library(name).map_err(|e| e.to_string())
250 }
251
252 /// Simple shortcut for using all default options for finding a library.
probe_library(name: &str) -> Result<Library, Error>253 pub fn probe_library(name: &str) -> Result<Library, Error> {
254 Config::new().probe(name)
255 }
256
257 #[doc(hidden)]
258 #[deprecated(note = "use config.target_supported() instance method instead")]
target_supported() -> bool259 pub fn target_supported() -> bool {
260 Config::new().target_supported()
261 }
262
263 /// Run `pkg-config` to get the value of a variable from a package using
264 /// `--variable`.
265 ///
266 /// The content of `PKG_CONFIG_SYSROOT_DIR` is not injected in paths that are
267 /// returned by `pkg-config --variable`, which makes them unsuitable to use
268 /// during cross-compilation unless specifically designed to be used
269 /// at that time.
get_variable(package: &str, variable: &str) -> Result<String, Error>270 pub fn get_variable(package: &str, variable: &str) -> Result<String, Error> {
271 let arg = format!("--variable={}", variable);
272 let cfg = Config::new();
273 let out = run(cfg.command(package, &[&arg]))?;
274 Ok(str::from_utf8(&out).unwrap().trim_end().to_owned())
275 }
276
277 impl Config {
278 /// Creates a new set of configuration options which are all initially set
279 /// to "blank".
new() -> Config280 pub fn new() -> Config {
281 Config {
282 statik: None,
283 min_version: Bound::Unbounded,
284 max_version: Bound::Unbounded,
285 extra_args: vec![],
286 print_system_cflags: true,
287 print_system_libs: true,
288 cargo_metadata: true,
289 env_metadata: true,
290 }
291 }
292
293 /// Indicate whether the `--static` flag should be passed.
294 ///
295 /// This will override the inference from environment variables described in
296 /// the crate documentation.
statik(&mut self, statik: bool) -> &mut Config297 pub fn statik(&mut self, statik: bool) -> &mut Config {
298 self.statik = Some(statik);
299 self
300 }
301
302 /// Indicate that the library must be at least version `vers`.
atleast_version(&mut self, vers: &str) -> &mut Config303 pub fn atleast_version(&mut self, vers: &str) -> &mut Config {
304 self.min_version = Bound::Included(vers.to_string());
305 self.max_version = Bound::Unbounded;
306 self
307 }
308
309 /// Indicate that the library must be equal to version `vers`.
exactly_version(&mut self, vers: &str) -> &mut Config310 pub fn exactly_version(&mut self, vers: &str) -> &mut Config {
311 self.min_version = Bound::Included(vers.to_string());
312 self.max_version = Bound::Included(vers.to_string());
313 self
314 }
315
316 /// Indicate that the library's version must be in `range`.
range_version<'a, R>(&mut self, range: R) -> &mut Config where R: RangeBounds<&'a str>,317 pub fn range_version<'a, R>(&mut self, range: R) -> &mut Config
318 where
319 R: RangeBounds<&'a str>,
320 {
321 self.min_version = match range.start_bound() {
322 Bound::Included(vers) => Bound::Included(vers.to_string()),
323 Bound::Excluded(vers) => Bound::Excluded(vers.to_string()),
324 Bound::Unbounded => Bound::Unbounded,
325 };
326 self.max_version = match range.end_bound() {
327 Bound::Included(vers) => Bound::Included(vers.to_string()),
328 Bound::Excluded(vers) => Bound::Excluded(vers.to_string()),
329 Bound::Unbounded => Bound::Unbounded,
330 };
331 self
332 }
333
334 /// Add an argument to pass to pkg-config.
335 ///
336 /// It's placed after all of the arguments generated by this library.
arg<S: AsRef<OsStr>>(&mut self, arg: S) -> &mut Config337 pub fn arg<S: AsRef<OsStr>>(&mut self, arg: S) -> &mut Config {
338 self.extra_args.push(arg.as_ref().to_os_string());
339 self
340 }
341
342 /// Define whether metadata should be emitted for cargo allowing it to
343 /// automatically link the binary. Defaults to `true`.
cargo_metadata(&mut self, cargo_metadata: bool) -> &mut Config344 pub fn cargo_metadata(&mut self, cargo_metadata: bool) -> &mut Config {
345 self.cargo_metadata = cargo_metadata;
346 self
347 }
348
349 /// Define whether metadata should be emitted for cargo allowing to
350 /// automatically rebuild when environment variables change. Defaults to
351 /// `true`.
env_metadata(&mut self, env_metadata: bool) -> &mut Config352 pub fn env_metadata(&mut self, env_metadata: bool) -> &mut Config {
353 self.env_metadata = env_metadata;
354 self
355 }
356
357 /// Enable or disable the `PKG_CONFIG_ALLOW_SYSTEM_LIBS` environment
358 /// variable.
359 ///
360 /// This env var is enabled by default.
print_system_libs(&mut self, print: bool) -> &mut Config361 pub fn print_system_libs(&mut self, print: bool) -> &mut Config {
362 self.print_system_libs = print;
363 self
364 }
365
366 /// Enable or disable the `PKG_CONFIG_ALLOW_SYSTEM_CFLAGS` environment
367 /// variable.
368 ///
369 /// This env var is enabled by default.
print_system_cflags(&mut self, print: bool) -> &mut Config370 pub fn print_system_cflags(&mut self, print: bool) -> &mut Config {
371 self.print_system_cflags = print;
372 self
373 }
374
375 /// Deprecated in favor fo the `probe` function
376 #[doc(hidden)]
find(&self, name: &str) -> Result<Library, String>377 pub fn find(&self, name: &str) -> Result<Library, String> {
378 self.probe(name).map_err(|e| e.to_string())
379 }
380
381 /// Run `pkg-config` to find the library `name`.
382 ///
383 /// This will use all configuration previously set to specify how
384 /// `pkg-config` is run.
probe(&self, name: &str) -> Result<Library, Error>385 pub fn probe(&self, name: &str) -> Result<Library, Error> {
386 let abort_var_name = format!("{}_NO_PKG_CONFIG", envify(name));
387 if self.env_var_os(&abort_var_name).is_some() {
388 return Err(Error::EnvNoPkgConfig(abort_var_name));
389 } else if !self.target_supported() {
390 return Err(Error::CrossCompilation);
391 }
392
393 let mut library = Library::new();
394
395 let output = run(self.command(name, &["--libs", "--cflags"])).map_err(|e| match e {
396 Error::Failure { command, output } => Error::ProbeFailure {
397 name: name.to_owned(),
398 command,
399 output,
400 },
401 other => other,
402 })?;
403 library.parse_libs_cflags(name, &output, self);
404
405 let output = run(self.command(name, &["--modversion"]))?;
406 library.parse_modversion(str::from_utf8(&output).unwrap());
407
408 Ok(library)
409 }
410
411 /// True if pkg-config is used for the host system, or configured for cross-compilation
target_supported(&self) -> bool412 pub fn target_supported(&self) -> bool {
413 let target = env::var_os("TARGET").unwrap_or_default();
414 let host = env::var_os("HOST").unwrap_or_default();
415
416 // Only use pkg-config in host == target situations by default (allowing an
417 // override).
418 if host == target {
419 return true;
420 }
421
422 // pkg-config may not be aware of cross-compilation, and require
423 // a wrapper script that sets up platform-specific prefixes.
424 match self.targetted_env_var("PKG_CONFIG_ALLOW_CROSS") {
425 // don't use pkg-config if explicitly disabled
426 Some(ref val) if val == "0" => false,
427 Some(_) => true,
428 None => {
429 // if not disabled, and pkg-config is customized,
430 // then assume it's prepared for cross-compilation
431 self.targetted_env_var("PKG_CONFIG").is_some()
432 || self.targetted_env_var("PKG_CONFIG_SYSROOT_DIR").is_some()
433 }
434 }
435 }
436
437 /// Deprecated in favor of the top level `get_variable` function
438 #[doc(hidden)]
get_variable(package: &str, variable: &str) -> Result<String, String>439 pub fn get_variable(package: &str, variable: &str) -> Result<String, String> {
440 get_variable(package, variable).map_err(|e| e.to_string())
441 }
442
targetted_env_var(&self, var_base: &str) -> Option<OsString>443 fn targetted_env_var(&self, var_base: &str) -> Option<OsString> {
444 match (env::var("TARGET"), env::var("HOST")) {
445 (Ok(target), Ok(host)) => {
446 let kind = if host == target { "HOST" } else { "TARGET" };
447 let target_u = target.replace("-", "_");
448
449 self.env_var_os(&format!("{}_{}", var_base, target))
450 .or_else(|| self.env_var_os(&format!("{}_{}", var_base, target_u)))
451 .or_else(|| self.env_var_os(&format!("{}_{}", kind, var_base)))
452 .or_else(|| self.env_var_os(var_base))
453 }
454 (Err(env::VarError::NotPresent), _) | (_, Err(env::VarError::NotPresent)) => {
455 self.env_var_os(var_base)
456 }
457 (Err(env::VarError::NotUnicode(s)), _) | (_, Err(env::VarError::NotUnicode(s))) => {
458 panic!(
459 "HOST or TARGET environment variable is not valid unicode: {:?}",
460 s
461 )
462 }
463 }
464 }
465
env_var_os(&self, name: &str) -> Option<OsString>466 fn env_var_os(&self, name: &str) -> Option<OsString> {
467 if self.env_metadata {
468 println!("cargo:rerun-if-env-changed={}", name);
469 }
470 env::var_os(name)
471 }
472
is_static(&self, name: &str) -> bool473 fn is_static(&self, name: &str) -> bool {
474 self.statik.unwrap_or_else(|| self.infer_static(name))
475 }
476
command(&self, name: &str, args: &[&str]) -> Command477 fn command(&self, name: &str, args: &[&str]) -> Command {
478 let exe = self
479 .targetted_env_var("PKG_CONFIG")
480 .unwrap_or_else(|| OsString::from("pkg-config"));
481 let mut cmd = Command::new(exe);
482 if self.is_static(name) {
483 cmd.arg("--static");
484 }
485 cmd.args(args).args(&self.extra_args);
486
487 if let Some(value) = self.targetted_env_var("PKG_CONFIG_PATH") {
488 cmd.env("PKG_CONFIG_PATH", value);
489 }
490 if let Some(value) = self.targetted_env_var("PKG_CONFIG_LIBDIR") {
491 cmd.env("PKG_CONFIG_LIBDIR", value);
492 }
493 if let Some(value) = self.targetted_env_var("PKG_CONFIG_SYSROOT_DIR") {
494 cmd.env("PKG_CONFIG_SYSROOT_DIR", value);
495 }
496 if self.print_system_libs {
497 cmd.env("PKG_CONFIG_ALLOW_SYSTEM_LIBS", "1");
498 }
499 if self.print_system_cflags {
500 cmd.env("PKG_CONFIG_ALLOW_SYSTEM_CFLAGS", "1");
501 }
502 cmd.arg(name);
503 match self.min_version {
504 Bound::Included(ref version) => {
505 cmd.arg(&format!("{} >= {}", name, version));
506 }
507 Bound::Excluded(ref version) => {
508 cmd.arg(&format!("{} > {}", name, version));
509 }
510 _ => (),
511 }
512 match self.max_version {
513 Bound::Included(ref version) => {
514 cmd.arg(&format!("{} <= {}", name, version));
515 }
516 Bound::Excluded(ref version) => {
517 cmd.arg(&format!("{} < {}", name, version));
518 }
519 _ => (),
520 }
521 cmd
522 }
523
print_metadata(&self, s: &str)524 fn print_metadata(&self, s: &str) {
525 if self.cargo_metadata {
526 println!("cargo:{}", s);
527 }
528 }
529
infer_static(&self, name: &str) -> bool530 fn infer_static(&self, name: &str) -> bool {
531 let name = envify(name);
532 if self.env_var_os(&format!("{}_STATIC", name)).is_some() {
533 true
534 } else if self.env_var_os(&format!("{}_DYNAMIC", name)).is_some() {
535 false
536 } else if self.env_var_os("PKG_CONFIG_ALL_STATIC").is_some() {
537 true
538 } else if self.env_var_os("PKG_CONFIG_ALL_DYNAMIC").is_some() {
539 false
540 } else {
541 false
542 }
543 }
544 }
545
546 // Implement Default manualy since Bound does not implement Default.
547 impl Default for Config {
default() -> Config548 fn default() -> Config {
549 Config {
550 statik: None,
551 min_version: Bound::Unbounded,
552 max_version: Bound::Unbounded,
553 extra_args: vec![],
554 print_system_cflags: false,
555 print_system_libs: false,
556 cargo_metadata: false,
557 env_metadata: false,
558 }
559 }
560 }
561
562 impl Library {
new() -> Library563 fn new() -> Library {
564 Library {
565 libs: Vec::new(),
566 link_paths: Vec::new(),
567 link_files: Vec::new(),
568 include_paths: Vec::new(),
569 ld_args: Vec::new(),
570 frameworks: Vec::new(),
571 framework_paths: Vec::new(),
572 defines: HashMap::new(),
573 version: String::new(),
574 _priv: (),
575 }
576 }
577
578 /// Extract the &str to pass to cargo:rustc-link-lib from a filename (just the file name, not including directories)
579 /// using target-specific logic.
extract_lib_from_filename<'a>(target: &str, filename: &'a str) -> Option<&'a str>580 fn extract_lib_from_filename<'a>(target: &str, filename: &'a str) -> Option<&'a str> {
581 fn test_suffixes<'b>(filename: &'b str, suffixes: &[&str]) -> Option<&'b str> {
582 for suffix in suffixes {
583 if filename.ends_with(suffix) {
584 return Some(&filename[..filename.len() - suffix.len()]);
585 }
586 }
587 None
588 }
589
590 let prefix = "lib";
591 if target.contains("msvc") {
592 // According to link.exe documentation:
593 // https://learn.microsoft.com/en-us/cpp/build/reference/link-input-files?view=msvc-170
594 //
595 // LINK doesn't use file extensions to make assumptions about the contents of a file.
596 // Instead, LINK examines each input file to determine what kind of file it is.
597 //
598 // However, rustc appends `.lib` to the string it receives from the -l command line argument,
599 // which it receives from Cargo via cargo:rustc-link-lib:
600 // https://github.com/rust-lang/rust/blob/657f246812ab2684e3c3954b1c77f98fd59e0b21/compiler/rustc_codegen_ssa/src/back/linker.rs#L828
601 // https://github.com/rust-lang/rust/blob/657f246812ab2684e3c3954b1c77f98fd59e0b21/compiler/rustc_codegen_ssa/src/back/linker.rs#L843
602 // So the only file extension that works for MSVC targets is `.lib`
603 return test_suffixes(filename, &[".lib"]);
604 } else if target.contains("windows") && target.contains("gnu") {
605 // GNU targets for Windows, including gnullvm, use `LinkerFlavor::Gcc` internally in rustc,
606 // which tells rustc to use the GNU linker. rustc does not prepend/append to the string it
607 // receives via the -l command line argument before passing it to the linker:
608 // https://github.com/rust-lang/rust/blob/657f246812ab2684e3c3954b1c77f98fd59e0b21/compiler/rustc_codegen_ssa/src/back/linker.rs#L446
609 // https://github.com/rust-lang/rust/blob/657f246812ab2684e3c3954b1c77f98fd59e0b21/compiler/rustc_codegen_ssa/src/back/linker.rs#L457
610 // GNU ld can work with more types of files than just the .lib files that MSVC's link.exe needs.
611 // GNU ld will prepend the `lib` prefix to the filename if necessary, so it is okay to remove
612 // the `lib` prefix from the filename. The `.a` suffix *requires* the `lib` prefix.
613 // https://sourceware.org/binutils/docs-2.39/ld.html#index-direct-linking-to-a-dll
614 if filename.starts_with(prefix) {
615 let filename = &filename[prefix.len()..];
616 return test_suffixes(filename, &[".dll.a", ".dll", ".lib", ".a"]);
617 } else {
618 return test_suffixes(filename, &[".dll.a", ".dll", ".lib"]);
619 }
620 } else if target.contains("apple") {
621 if filename.starts_with(prefix) {
622 let filename = &filename[prefix.len()..];
623 return test_suffixes(filename, &[".a", ".so", ".dylib"]);
624 }
625 return None;
626 } else {
627 if filename.starts_with(prefix) {
628 let filename = &filename[prefix.len()..];
629 return test_suffixes(filename, &[".a", ".so"]);
630 }
631 return None;
632 }
633 }
634
parse_libs_cflags(&mut self, name: &str, output: &[u8], config: &Config)635 fn parse_libs_cflags(&mut self, name: &str, output: &[u8], config: &Config) {
636 let mut is_msvc = false;
637 let target = env::var("TARGET");
638 if let Ok(target) = &target {
639 if target.contains("msvc") {
640 is_msvc = true;
641 }
642 }
643
644 let system_roots = if cfg!(target_os = "macos") {
645 vec![PathBuf::from("/Library"), PathBuf::from("/System")]
646 } else {
647 let sysroot = config
648 .env_var_os("PKG_CONFIG_SYSROOT_DIR")
649 .or_else(|| config.env_var_os("SYSROOT"))
650 .map(PathBuf::from);
651
652 if cfg!(target_os = "windows") {
653 if let Some(sysroot) = sysroot {
654 vec![sysroot]
655 } else {
656 vec![]
657 }
658 } else {
659 vec![sysroot.unwrap_or_else(|| PathBuf::from("/usr"))]
660 }
661 };
662
663 let mut dirs = Vec::new();
664 let statik = config.is_static(name);
665
666 let words = split_flags(output);
667
668 // Handle single-character arguments like `-I/usr/include`
669 let parts = words
670 .iter()
671 .filter(|l| l.len() > 2)
672 .map(|arg| (&arg[0..2], &arg[2..]));
673 for (flag, val) in parts {
674 match flag {
675 "-L" => {
676 let meta = format!("rustc-link-search=native={}", val);
677 config.print_metadata(&meta);
678 dirs.push(PathBuf::from(val));
679 self.link_paths.push(PathBuf::from(val));
680 }
681 "-F" => {
682 let meta = format!("rustc-link-search=framework={}", val);
683 config.print_metadata(&meta);
684 self.framework_paths.push(PathBuf::from(val));
685 }
686 "-I" => {
687 self.include_paths.push(PathBuf::from(val));
688 }
689 "-l" => {
690 // These are provided by the CRT with MSVC
691 if is_msvc && ["m", "c", "pthread"].contains(&val) {
692 continue;
693 }
694
695 if statik && is_static_available(val, &system_roots, &dirs) {
696 let meta = format!("rustc-link-lib=static={}", val);
697 config.print_metadata(&meta);
698 } else {
699 let meta = format!("rustc-link-lib={}", val);
700 config.print_metadata(&meta);
701 }
702
703 self.libs.push(val.to_string());
704 }
705 "-D" => {
706 let mut iter = val.split('=');
707 self.defines.insert(
708 iter.next().unwrap().to_owned(),
709 iter.next().map(|s| s.to_owned()),
710 );
711 }
712 _ => {}
713 }
714 }
715
716 // Handle multi-character arguments with space-separated value like `-framework foo`
717 let mut iter = words.iter().flat_map(|arg| {
718 if arg.starts_with("-Wl,") {
719 arg[4..].split(',').collect()
720 } else {
721 vec![arg.as_ref()]
722 }
723 });
724 while let Some(part) = iter.next() {
725 match part {
726 "-framework" => {
727 if let Some(lib) = iter.next() {
728 let meta = format!("rustc-link-lib=framework={}", lib);
729 config.print_metadata(&meta);
730 self.frameworks.push(lib.to_string());
731 }
732 }
733 "-isystem" | "-iquote" | "-idirafter" => {
734 if let Some(inc) = iter.next() {
735 self.include_paths.push(PathBuf::from(inc));
736 }
737 }
738 _ => {
739 let path = std::path::Path::new(part);
740 if path.is_file() {
741 // Cargo doesn't have a means to directly specify a file path to link,
742 // so split up the path into the parent directory and library name.
743 // TODO: pass file path directly when link-arg library type is stabilized
744 // https://github.com/rust-lang/rust/issues/99427
745 if let (Some(dir), Some(file_name), Ok(target)) =
746 (path.parent(), path.file_name(), &target)
747 {
748 match Self::extract_lib_from_filename(
749 target,
750 &file_name.to_string_lossy(),
751 ) {
752 Some(lib_basename) => {
753 let link_search =
754 format!("rustc-link-search={}", dir.display());
755 config.print_metadata(&link_search);
756
757 let link_lib = format!("rustc-link-lib={}", lib_basename);
758 config.print_metadata(&link_lib);
759 self.link_files.push(PathBuf::from(path));
760 }
761 None => {
762 println!("cargo:warning=File path {} found in pkg-config file for {}, but could not extract library base name to pass to linker command line", path.display(), name);
763 }
764 }
765 }
766 }
767 }
768 }
769 }
770
771 let mut linker_options = words.iter().filter(|arg| arg.starts_with("-Wl,"));
772 while let Some(option) = linker_options.next() {
773 let mut pop = false;
774 let mut ld_option = vec![];
775 for subopt in option[4..].split(',') {
776 if pop {
777 pop = false;
778 continue;
779 }
780
781 if subopt == "-framework" {
782 pop = true;
783 continue;
784 }
785
786 ld_option.push(subopt);
787 }
788
789 let meta = format!("rustc-link-arg=-Wl,{}", ld_option.join(","));
790 config.print_metadata(&meta);
791
792 self.ld_args
793 .push(ld_option.into_iter().map(String::from).collect());
794 }
795 }
796
parse_modversion(&mut self, output: &str)797 fn parse_modversion(&mut self, output: &str) {
798 self.version.push_str(output.lines().nth(0).unwrap().trim());
799 }
800 }
801
envify(name: &str) -> String802 fn envify(name: &str) -> String {
803 name.chars()
804 .map(|c| c.to_ascii_uppercase())
805 .map(|c| if c == '-' { '_' } else { c })
806 .collect()
807 }
808
809 /// System libraries should only be linked dynamically
is_static_available(name: &str, system_roots: &[PathBuf], dirs: &[PathBuf]) -> bool810 fn is_static_available(name: &str, system_roots: &[PathBuf], dirs: &[PathBuf]) -> bool {
811 let libname = format!("lib{}.a", name);
812
813 dirs.iter().any(|dir| {
814 !system_roots.iter().any(|sys| dir.starts_with(sys)) && dir.join(&libname).exists()
815 })
816 }
817
run(mut cmd: Command) -> Result<Vec<u8>, Error>818 fn run(mut cmd: Command) -> Result<Vec<u8>, Error> {
819 match cmd.output() {
820 Ok(output) => {
821 if output.status.success() {
822 Ok(output.stdout)
823 } else {
824 Err(Error::Failure {
825 command: format!("{:?}", cmd),
826 output,
827 })
828 }
829 }
830 Err(cause) => Err(Error::Command {
831 command: format!("{:?}", cmd),
832 cause,
833 }),
834 }
835 }
836
837 /// Split output produced by pkg-config --cflags and / or --libs into separate flags.
838 ///
839 /// Backslash in output is used to preserve literal meaning of following byte. Different words are
840 /// separated by unescaped space. Other whitespace characters generally should not occur unescaped
841 /// at all, apart from the newline at the end of output. For compatibility with what others
842 /// consumers of pkg-config output would do in this scenario, they are used here for splitting as
843 /// well.
split_flags(output: &[u8]) -> Vec<String>844 fn split_flags(output: &[u8]) -> Vec<String> {
845 let mut word = Vec::new();
846 let mut words = Vec::new();
847 let mut escaped = false;
848
849 for &b in output {
850 match b {
851 _ if escaped => {
852 escaped = false;
853 word.push(b);
854 }
855 b'\\' => escaped = true,
856 b'\t' | b'\n' | b'\r' | b' ' => {
857 if !word.is_empty() {
858 words.push(String::from_utf8(word).unwrap());
859 word = Vec::new();
860 }
861 }
862 _ => word.push(b),
863 }
864 }
865
866 if !word.is_empty() {
867 words.push(String::from_utf8(word).unwrap());
868 }
869
870 words
871 }
872
873 #[cfg(test)]
874 mod tests {
875 use super::*;
876
877 #[test]
878 #[cfg(target_os = "macos")]
system_library_mac_test()879 fn system_library_mac_test() {
880 use std::path::Path;
881
882 let system_roots = vec![PathBuf::from("/Library"), PathBuf::from("/System")];
883
884 assert!(!is_static_available(
885 "PluginManager",
886 &system_roots,
887 &[PathBuf::from("/Library/Frameworks")]
888 ));
889 assert!(!is_static_available(
890 "python2.7",
891 &system_roots,
892 &[PathBuf::from(
893 "/System/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/config"
894 )]
895 ));
896 assert!(!is_static_available(
897 "ffi_convenience",
898 &system_roots,
899 &[PathBuf::from(
900 "/Library/Ruby/Gems/2.0.0/gems/ffi-1.9.10/ext/ffi_c/libffi-x86_64/.libs"
901 )]
902 ));
903
904 // Homebrew is in /usr/local, and it's not a part of the OS
905 if Path::new("/usr/local/lib/libpng16.a").exists() {
906 assert!(is_static_available(
907 "png16",
908 &system_roots,
909 &[PathBuf::from("/usr/local/lib")]
910 ));
911
912 let libpng = Config::new()
913 .range_version("1".."99")
914 .probe("libpng16")
915 .unwrap();
916 assert!(libpng.version.find('\n').is_none());
917 }
918 }
919
920 #[test]
921 #[cfg(target_os = "linux")]
system_library_linux_test()922 fn system_library_linux_test() {
923 assert!(!is_static_available(
924 "util",
925 &[PathBuf::from("/usr")],
926 &[PathBuf::from("/usr/lib/x86_64-linux-gnu")]
927 ));
928 assert!(!is_static_available(
929 "dialog",
930 &[PathBuf::from("/usr")],
931 &[PathBuf::from("/usr/lib")]
932 ));
933 }
934
test_library_filename(target: &str, filename: &str)935 fn test_library_filename(target: &str, filename: &str) {
936 assert_eq!(
937 Library::extract_lib_from_filename(target, filename),
938 Some("foo")
939 );
940 }
941
942 #[test]
link_filename_linux()943 fn link_filename_linux() {
944 let target = "x86_64-unknown-linux-gnu";
945 test_library_filename(target, "libfoo.a");
946 test_library_filename(target, "libfoo.so");
947 }
948
949 #[test]
link_filename_apple()950 fn link_filename_apple() {
951 let target = "x86_64-apple-darwin";
952 test_library_filename(target, "libfoo.a");
953 test_library_filename(target, "libfoo.so");
954 test_library_filename(target, "libfoo.dylib");
955 }
956
957 #[test]
link_filename_msvc()958 fn link_filename_msvc() {
959 let target = "x86_64-pc-windows-msvc";
960 // static and dynamic libraries have the same .lib suffix
961 test_library_filename(target, "foo.lib");
962 }
963
964 #[test]
link_filename_mingw()965 fn link_filename_mingw() {
966 let target = "x86_64-pc-windows-gnu";
967 test_library_filename(target, "foo.lib");
968 test_library_filename(target, "libfoo.a");
969 test_library_filename(target, "foo.dll");
970 test_library_filename(target, "foo.dll.a");
971 }
972 }
973