| // SPDX-License-Identifier: Apache-2.0 | |
| use std::path::{Path, PathBuf}; | |
| use glob::Pattern; | |
| use super::common; | |
| //================================================ | |
| // Searching | |
| //================================================ | |
| /// Clang static libraries required to link to `libclang` 3.5 and later. | |
| const CLANG_LIBRARIES: &[&str] = &[ | |
| "clang", | |
| "clangAST", | |
| "clangAnalysis", | |
| "clangBasic", | |
| "clangDriver", | |
| "clangEdit", | |
| "clangFrontend", | |
| "clangIndex", | |
| "clangLex", | |
| "clangParse", | |
| "clangRewrite", | |
| "clangSema", | |
| "clangSerialization", | |
| ]; | |
| /// Gets the name of an LLVM or Clang static library from a path. | |
| fn get_library_name(path: &Path) -> Option<String> { | |
| path.file_stem().map(|p| { | |
| let string = p.to_string_lossy(); | |
| if let Some(name) = string.strip_prefix("lib") { | |
| name.to_owned() | |
| } else { | |
| string.to_string() | |
| } | |
| }) | |
| } | |
| /// Gets the LLVM static libraries required to link to `libclang`. | |
| fn get_llvm_libraries() -> Vec<String> { | |
| common::run_llvm_config(&["--libs"]) | |
| .unwrap() | |
| .split_whitespace() | |
| .filter_map(|p| { | |
| // Depending on the version of `llvm-config` in use, listed | |
| // libraries may be in one of two forms, a full path to the library | |
| // or simply prefixed with `-l`. | |
| if let Some(path) = p.strip_prefix("-l") { | |
| Some(path.into()) | |
| } else { | |
| get_library_name(Path::new(p)) | |
| } | |
| }) | |
| .collect() | |
| } | |
| /// Gets the Clang static libraries required to link to `libclang`. | |
| fn get_clang_libraries<P: AsRef<Path>>(directory: P) -> Vec<String> { | |
| // Escape the directory in case it contains characters that have special | |
| // meaning in glob patterns (e.g., `[` or `]`). | |
| let directory = Pattern::escape(directory.as_ref().to_str().unwrap()); | |
| let directory = Path::new(&directory); | |
| let pattern = directory.join("libclang*.a").to_str().unwrap().to_owned(); | |
| if let Ok(libraries) = glob::glob(&pattern) { | |
| libraries | |
| .filter_map(|l| l.ok().and_then(|l| get_library_name(&l))) | |
| .collect() | |
| } else { | |
| CLANG_LIBRARIES.iter().map(|l| (*l).to_string()).collect() | |
| } | |
| } | |
| /// Finds a directory containing LLVM and Clang static libraries and returns the | |
| /// path to that directory. | |
| fn find() -> PathBuf { | |
| let name = if target_os!("windows") { | |
| "libclang.lib" | |
| } else { | |
| "libclang.a" | |
| }; | |
| let files = common::search_libclang_directories(&[name.into()], "LIBCLANG_STATIC_PATH"); | |
| if let Some((directory, _)) = files.into_iter().next() { | |
| directory | |
| } else { | |
| panic!( | |
| "could not find the required `{name}` static library, see the \ | |
| README for more information on how to link to `libclang` statically: \ | |
| https://github.com/KyleMayes/clang-sys?tab=readme-ov-file#static" | |
| ); | |
| } | |
| } | |
| //================================================ | |
| // Linking | |
| //================================================ | |
| /// Finds and links to `libclang` static libraries. | |
| pub fn link() { | |
| let cep = common::CommandErrorPrinter::default(); | |
| let directory = find(); | |
| // Specify required Clang static libraries. | |
| println!("cargo:rustc-link-search=native={}", directory.display()); | |
| for library in get_clang_libraries(directory) { | |
| println!("cargo:rustc-link-lib=static={}", library); | |
| } | |
| // Determine the shared mode used by LLVM. | |
| let mode = common::run_llvm_config(&["--shared-mode"]).map(|m| m.trim().to_owned()); | |
| let prefix = if mode.map_or(false, |m| m == "static") { | |
| "static=" | |
| } else { | |
| "" | |
| }; | |
| // Specify required LLVM static libraries. | |
| println!( | |
| "cargo:rustc-link-search=native={}", | |
| common::run_llvm_config(&["--libdir"]).unwrap().trim_end() | |
| ); | |
| for library in get_llvm_libraries() { | |
| println!("cargo:rustc-link-lib={}{}", prefix, library); | |
| } | |
| // Specify required system libraries. | |
| // MSVC doesn't need this, as it tracks dependencies inside `.lib` files. | |
| if cfg!(target_os = "freebsd") { | |
| println!("cargo:rustc-flags=-l ffi -l ncursesw -l c++ -l z"); | |
| } else if cfg!(any(target_os = "haiku", target_os = "linux")) { | |
| if cfg!(feature = "libcpp") { | |
| println!("cargo:rustc-flags=-l c++"); | |
| } else { | |
| println!("cargo:rustc-flags=-l ffi -l ncursesw -l stdc++ -l z"); | |
| } | |
| } else if cfg!(target_os = "macos") { | |
| println!("cargo:rustc-flags=-l ffi -l ncurses -l c++ -l z"); | |
| } | |
| cep.discard(); | |
| } |