// Copyright 2018 Kyle Mayes // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. extern crate glob; use std::path::{Path, PathBuf}; use common; /// Returns the name of an LLVM or Clang library from a path to such a library. fn get_library_name(path: &Path) -> Option { path.file_stem().map(|p| { let string = p.to_string_lossy(); if string.starts_with("lib") { string[3..].to_owned() } else { string.to_string() } }) } /// Returns the LLVM libraries required to link to `libclang` statically. fn get_llvm_libraries() -> Vec { 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 p.starts_with("-l") { Some(p[2..].into()) } else { get_library_name(Path::new(p)) } }) .collect() } /// Clang libraries required to link to `libclang` 3.5 and later statically. const CLANG_LIBRARIES: &[&str] = &[ "clang", "clangAST", "clangAnalysis", "clangBasic", "clangDriver", "clangEdit", "clangFrontend", "clangIndex", "clangLex", "clangParse", "clangRewrite", "clangSema", "clangSerialization", ]; /// Returns the Clang libraries required to link to `libclang` statically. fn get_clang_libraries>(directory: P) -> Vec { let pattern = directory .as_ref() .join("libclang*.a") .to_string_lossy() .to_string(); 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() } } /// Returns a directory containing `libclang` static libraries. fn find() -> PathBuf { let name = if cfg!(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 any static libraries"); } } /// Find and link to `libclang` statically. 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!(target_os = "linux") { 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(); }