forked from coreylowman/dfdx
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbuild.rs
219 lines (188 loc) · 7.4 KB
/
build.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
//! This script links to Intel MKL when the `intel-mkl` feature is enabled.
//! The dynamically linked and threaded implementation is chosen as
//! a good default, because:
//! 1. Dynamic compiles faster and makes smaller binaries
//! 2. Threaded implementation is faster for large workloads.
//!
//! As described [by Intel here](https://www.intel.com/content/www/us/en/developer/articles/technical/intel-math-kernel-library-intel-mkl-and-pkg-config-tool.html).
//!
//! **NOTE** lp64 vs ilp64 is chosen based on [target_pointer_width](https://doc.rust-lang.org/reference/conditional-compilation.html#target_pointer_width) cfg variable.
//!
//! # Adding a new system
//!
//! To add a new target system, the following blocks need to exist:
//!
//! - `SHARED_LIB_DIRS`: This should contain directories relative to `ONEAPI_ROOT` environment variable that contain the
//! shared libraries. `main()` will check if any of these directories are not on the `PATH` environment variable and crash if not.
//! - `LINK_DIRS`: The directory where `.lib` files are. `main()` will output a `cargo:rustc-link-search` for each of these.
//!
//! # Supported systems
//!
//! - [x] Windows 32 bit
//! - [x] Windows 64 bit
//! - [x] Linux 32 bit
//! - [x] Linux 64 bit
//! - [x] MacOS 64 bit
//!
//! This script also creates a "nightly" feature if the crate is compiled on a nightly branch
use rustc_version::{version_meta, Channel};
pub const MKL_VERSION: &str = "2022.1.0";
pub const STATIC: bool = false;
pub const DYNAMIC: bool = true;
pub const SEQUENTIAL: bool = false;
pub const THREADED: bool = true;
pub const MKL: bool = (STATIC || DYNAMIC) && (SEQUENTIAL || THREADED);
pub const LINK_TYPE: &str = if STATIC { "static" } else { "dylib" };
pub const LIB_POSTFIX: &str = if cfg!(windows) && DYNAMIC { "_dll" } else { "" };
pub const LD_DIR: &str = if cfg!(windows) {
"PATH"
} else if cfg!(target_os = "linux") {
"LD_LIBRARY_PATH"
} else if cfg!(target_os = "macos") {
"DYLD_LIBRARY_PATH"
} else {
""
};
pub const DEFAULT_ONEAPI_ROOT: &str = if cfg!(windows) {
"C:/Program Files (x86)/Intel/oneAPI/"
} else {
"/opt/intel/oneapi/"
};
pub const MKL_CORE: &str = "mkl_core";
pub const MKL_THREAD: &str = if SEQUENTIAL {
"mkl_sequential"
} else {
"mkl_intel_thread"
};
pub const THREADING_LIB: &str = if cfg!(windows) { "libiomp5md" } else { "iomp5" };
pub const MKL_INTERFACE: &str = if cfg!(target_pointer_width = "32") {
"mkl_intel_ilp64"
} else {
"mkl_intel_lp64"
};
#[cfg(not(any(target_os = "linux", target_os = "windows", target_os = "macos")))]
pub const UNSUPPORTED_OS_ERROR: _ = "Target OS is not supported. Please contact me";
#[cfg(all(target_os = "windows", target_pointer_width = "32"))]
pub const LINK_DIRS: &[&str] = &[
"compiler/latest/windows/compiler/lib/ia32_win",
"mkl/latest/lib/ia32",
];
#[cfg(all(target_os = "windows", target_pointer_width = "64"))]
pub const LINK_DIRS: &[&str] = &[
"compiler/latest/windows/compiler/lib/intel64_win",
"mkl/latest/lib/intel64",
];
#[cfg(all(target_os = "windows", target_pointer_width = "32"))]
pub const SHARED_LIB_DIRS: &[&str] = &[
"compiler/latest/windows/redist/ia32_win/compiler",
"mkl/latest/redist/ia32",
];
#[cfg(all(target_os = "windows", target_pointer_width = "64"))]
pub const SHARED_LIB_DIRS: &[&str] = &[
"compiler/latest/windows/redist/intel64_win/compiler",
"mkl/latest/redist/intel64",
];
#[cfg(all(target_os = "linux", target_pointer_width = "32"))]
pub const LINK_DIRS: &[&str] = &[
"compiler/latest/linux/compiler/lib/ia32_lin",
"mkl/latest/lib/ia32",
];
#[cfg(all(target_os = "linux", target_pointer_width = "64"))]
pub const LINK_DIRS: &[&str] = &[
"compiler/latest/linux/compiler/lib/intel64_lin",
"mkl/latest/lib/intel64",
];
#[cfg(target_os = "linux")]
pub const SHARED_LIB_DIRS: &[&str] = LINK_DIRS;
#[cfg(target_os = "macos")]
const MACOS_COMPILER_PATH: &str = "compiler/latest/mac/compiler/lib";
#[cfg(target_os = "macos")]
pub const LINK_DIRS: &[&str] = &[MACOS_COMPILER_PATH, "mkl/latest/lib"];
#[cfg(target_os = "macos")]
pub const SHARED_LIB_DIRS: &[&str] = &["mkl/latest/lib"];
#[derive(Debug)]
pub enum BuildError {
OneAPINotFound(std::path::PathBuf),
OneAPINotADir(std::path::PathBuf),
PathNotFound(std::env::VarError),
AddSharedLibDirToPath(String),
}
fn main() -> Result<(), BuildError> {
println!("cargo:rerun-if-changed=build.rs");
#[cfg(feature = "intel-mkl")]
{
let root = std::env::var("ONEAPI_ROOT").unwrap_or_else(|_| DEFAULT_ONEAPI_ROOT.to_string());
println!("Using '{root}' as ONEAPI_ROOT");
let path = match std::env::var(LD_DIR) {
Ok(path) => path,
Err(e) => {
// On macOS it's unusual to set $DYLD_LIBRARY_PATH, so we want to provide a helpful message
println!(
"Library path env variable '{LD_DIR}' was not found. Run `{}`",
suggest_setvars_cmd(&root)
);
return Err(BuildError::PathNotFound(e));
}
};
if DYNAMIC {
// check to make sure that things in `SHARED_LIB_DIRS` are in `$LD_DIR`.
let path = path.replace('\\', "/");
for shared_lib_dir in SHARED_LIB_DIRS {
let versioned_dir = shared_lib_dir.replace("latest", MKL_VERSION);
println!("Checking that '{shared_lib_dir}' or '{versioned_dir}' is in {LD_DIR}");
if !path.contains(shared_lib_dir) && !path.contains(&versioned_dir) {
println!(
"'{shared_lib_dir}' not found in library path. Run `{}`",
suggest_setvars_cmd(&root)
);
return Err(BuildError::AddSharedLibDirToPath(
shared_lib_dir.to_string(),
));
}
}
}
let root: std::path::PathBuf = root.into();
if !root.exists() {
return Err(BuildError::OneAPINotFound(root));
}
if !root.is_dir() {
return Err(BuildError::OneAPINotADir(root));
}
for rel_lib_dir in LINK_DIRS {
let lib_dir = root.join(rel_lib_dir);
println!("cargo:rustc-link-search={}", lib_dir.display());
}
println!("cargo:rustc-link-lib={LINK_TYPE}={MKL_INTERFACE}{LIB_POSTFIX}");
println!("cargo:rustc-link-lib={LINK_TYPE}={MKL_THREAD}{LIB_POSTFIX}");
println!("cargo:rustc-link-lib={LINK_TYPE}={MKL_CORE}{LIB_POSTFIX}");
if THREADED {
println!("cargo:rustc-link-lib=dylib={THREADING_LIB}");
}
if !cfg!(windows) {
println!("cargo:rustc-link-lib=pthread");
println!("cargo:rustc-link-lib=m");
println!("cargo:rustc-link-lib=dl");
}
#[cfg(target_os = "macos")]
{
println!(
"cargo:rustc-link-arg=-Wl,-rpath,{}/{MACOS_COMPILER_PATH}",
root.display(),
);
}
}
// If on nightly, enable "nightly" feature
if version_meta().unwrap().channel == Channel::Nightly {
println!("cargo:rustc-cfg=feature=\"nightly\"");
}
Ok(())
}
// This section creates a feature "nightly" enabled if built on a nightly branch
#[cfg(feature = "intel-mkl")]
fn suggest_setvars_cmd(root: &str) -> String {
if cfg!(windows) {
format!("{root}/setvars.bat")
} else {
format!("source {root}/setvars.sh")
}
}