182 lines
6.3 KiB
Rust
182 lines
6.3 KiB
Rust
#![allow(clippy::uninlined_format_args)]
|
|
|
|
extern crate bindgen;
|
|
|
|
use cfg_if::cfg_if;
|
|
use std::env;
|
|
use std::path::PathBuf;
|
|
|
|
fn main() {
|
|
let target = env::var("TARGET").unwrap();
|
|
// Link C++ standard library
|
|
if let Some(cpp_stdlib) = get_cpp_link_stdlib(&target) {
|
|
println!("cargo:rustc-link-lib=dylib={}", cpp_stdlib);
|
|
}
|
|
// Link macOS Accelerate framework for matrix calculations
|
|
if target.contains("apple") {
|
|
println!("cargo:rustc-link-lib=framework=Accelerate");
|
|
#[cfg(feature = "coreml")]
|
|
{
|
|
println!("cargo:rustc-link-lib=framework=Foundation");
|
|
println!("cargo:rustc-link-lib=framework=CoreML");
|
|
}
|
|
}
|
|
|
|
println!("cargo:rustc-link-search={}", env::var("OUT_DIR").unwrap());
|
|
println!("cargo:rustc-link-lib=static=whisper");
|
|
#[cfg(feature = "coreml")]
|
|
println!("cargo:rustc-link-lib=static=whisper.coreml");
|
|
#[cfg(feature = "opencl")]
|
|
{
|
|
println!("cargo:rustc-link-lib=clblast");
|
|
println!("cargo:rustc-link-lib=OpenCL");
|
|
}
|
|
#[cfg(feature = "cuda")]
|
|
{
|
|
println!("cargo:rustc-link-lib=cublas");
|
|
println!("cargo:rustc-link-lib=cudart");
|
|
println!("cargo:rustc-link-lib=cublasLt");
|
|
cfg_if! {
|
|
if #[cfg(target_os = "windows")] {
|
|
let cuda_path = PathBuf::from(env::var("CUDA_PATH").unwrap()).join("lib/x64");
|
|
println!("cargo:rustc-link-search={}", cuda_path.display());
|
|
} else {
|
|
println!("cargo:rustc-link-lib=culibos");
|
|
println!("cargo:rustc-link-search=/usr/local/cuda/lib64");
|
|
println!("cargo:rustc-link-search=/opt/cuda/lib64");
|
|
}
|
|
}
|
|
}
|
|
println!("cargo:rerun-if-changed=wrapper.h");
|
|
|
|
if env::var("WHISPER_DONT_GENERATE_BINDINGS").is_ok() {
|
|
let _: u64 = std::fs::copy(
|
|
"src/bindings.rs",
|
|
env::var("OUT_DIR").unwrap() + "/bindings.rs",
|
|
)
|
|
.expect("Failed to copy bindings.rs");
|
|
} else {
|
|
let bindings = bindgen::Builder::default()
|
|
.header("wrapper.h")
|
|
.clang_arg("-I./whisper.cpp")
|
|
.clang_arg("-I./ggml.c")
|
|
.parse_callbacks(Box::new(bindgen::CargoCallbacks))
|
|
.generate();
|
|
|
|
match bindings {
|
|
Ok(b) => {
|
|
let out_path = PathBuf::from(env::var("OUT_DIR").unwrap());
|
|
b.write_to_file(out_path.join("bindings.rs"))
|
|
.expect("Couldn't write bindings!");
|
|
}
|
|
Err(e) => {
|
|
println!("cargo:warning=Unable to generate bindings: {}", e);
|
|
println!("cargo:warning=Using bundled bindings.rs, which may be out of date");
|
|
// copy src/bindings.rs to OUT_DIR
|
|
std::fs::copy(
|
|
"src/bindings.rs",
|
|
env::var("OUT_DIR").unwrap() + "/bindings.rs",
|
|
)
|
|
.expect("Unable to copy bindings.rs");
|
|
}
|
|
}
|
|
};
|
|
|
|
// stop if we're on docs.rs
|
|
if env::var("DOCS_RS").is_ok() {
|
|
return;
|
|
}
|
|
|
|
// build libwhisper.a
|
|
env::set_current_dir("whisper.cpp").expect("Unable to change directory to whisper.cpp");
|
|
_ = std::fs::remove_dir_all("build");
|
|
_ = std::fs::create_dir("build");
|
|
env::set_current_dir("build").expect("Unable to change directory to whisper.cpp build");
|
|
|
|
let mut cmd = std::process::Command::new("cmake");
|
|
cmd.arg("..")
|
|
.arg("-DCMAKE_BUILD_TYPE=Release")
|
|
.arg("-DBUILD_SHARED_LIBS=OFF")
|
|
.arg("-DWHISPER_ALL_WARNINGS=OFF")
|
|
.arg("-DWHISPER_ALL_WARNINGS_3RD_PARTY=OFF")
|
|
.arg("-DWHISPER_BUILD_TESTS=OFF")
|
|
.arg("-DWHISPER_BUILD_EXAMPLES=OFF");
|
|
|
|
#[cfg(feature = "coreml")]
|
|
cmd.arg("-DWHISPER_COREML=ON")
|
|
.arg("-DWHISPER_COREML_ALLOW_FALLBACK=1");
|
|
|
|
#[cfg(feature = "cuda")]
|
|
cmd.arg("-DWHISPER_CUBLAS=ON");
|
|
|
|
#[cfg(feature = "opencl")]
|
|
cmd.arg("-DWHISPER_CLBLAST=ON");
|
|
|
|
cmd.arg("-DCMAKE_POSITION_INDEPENDENT_CODE=ON");
|
|
|
|
let code = cmd
|
|
.status()
|
|
.expect("Failed to run `cmake` (is CMake installed?)");
|
|
if code.code() != Some(0) {
|
|
panic!("Failed to run `cmake` (is CMake installed?)");
|
|
}
|
|
|
|
let code = std::process::Command::new("cmake")
|
|
.arg("--build")
|
|
.arg(".")
|
|
.arg("--config")
|
|
.arg("Release")
|
|
.status()
|
|
.expect("Failed to build libwhisper.a (is CMake installed?)");
|
|
if code.code() != Some(0) {
|
|
panic!("Failed to build libwhisper.a (is CMake installed?)");
|
|
}
|
|
|
|
// move libwhisper.a to where Cargo expects it (OUT_DIR)
|
|
cfg_if! {
|
|
if #[cfg(target_os = "windows")] {
|
|
std::fs::copy(
|
|
"Release/whisper.lib",
|
|
format!("{}/whisper.lib", env::var("OUT_DIR").unwrap()),
|
|
)
|
|
.expect("Failed to copy libwhisper.lib");
|
|
} else {
|
|
std::fs::copy(
|
|
"libwhisper.a",
|
|
format!("{}/libwhisper.a", env::var("OUT_DIR").unwrap()),
|
|
)
|
|
.expect("Failed to copy libwhisper.a");
|
|
}
|
|
}
|
|
|
|
// if on iOS or macOS, with coreml feature enabled, copy libwhisper.coreml.a as well
|
|
cfg_if! {
|
|
if #[cfg(all(feature = "coreml", any(target_os = "ios", target_os = "macos")))]
|
|
{
|
|
std::fs::copy(
|
|
"libwhisper.coreml.a",
|
|
format!("{}/libwhisper.coreml.a", env::var("OUT_DIR").unwrap()),
|
|
)
|
|
.expect("Failed to copy libwhisper.coreml.a");
|
|
}
|
|
}
|
|
|
|
// clean the whisper build directory to prevent Cargo from complaining during crate publish
|
|
env::set_current_dir("..").expect("Unable to change directory to whisper.cpp");
|
|
_ = std::fs::remove_dir_all("build");
|
|
// for whatever reason this file is generated during build and triggers cargo complaining
|
|
_ = std::fs::remove_file("bindings/javascript/package.json");
|
|
}
|
|
|
|
// From https://github.com/alexcrichton/cc-rs/blob/fba7feded71ee4f63cfe885673ead6d7b4f2f454/src/lib.rs#L2462
|
|
fn get_cpp_link_stdlib(target: &str) -> Option<&'static str> {
|
|
if target.contains("msvc") {
|
|
None
|
|
} else if target.contains("apple") || target.contains("freebsd") || target.contains("openbsd") {
|
|
Some("c++")
|
|
} else if target.contains("android") {
|
|
Some("c++_shared")
|
|
} else {
|
|
Some("stdc++")
|
|
}
|
|
}
|