1#![warn(missing_docs)]
10
11#[cfg(test)]
53#[macro_use]
54extern crate doc_comment;
55
56#[cfg(test)]
57doctest!("../README.md");
58
59use std::collections::HashMap;
60use std::process::Command;
61use std::{env, error, fmt, io, num, str};
62use std::{ffi::OsString, str::FromStr};
63
64pub use semver::Version;
67
68use Error::*;
69
70#[derive(Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Hash, Debug)]
72pub enum Channel {
73 Dev,
75 Nightly,
77 Beta,
79 Stable,
81}
82
83#[derive(Clone, Debug, Eq, PartialEq, Ord, PartialOrd, Hash)]
91pub struct LlvmVersion {
92 pub major: u64,
95 pub minor: u64,
97 }
99
100impl fmt::Display for LlvmVersion {
101 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
102 write!(f, "{}.{}", self.major, self.minor)
103 }
104}
105
106impl FromStr for LlvmVersion {
107 type Err = LlvmVersionParseError;
108
109 fn from_str(s: &str) -> Result<Self, Self::Err> {
110 let mut parts = s
111 .split('.')
112 .map(|part| -> Result<u64, LlvmVersionParseError> {
113 if part == "0" {
114 Ok(0)
115 } else if part.starts_with('0') {
116 Err(LlvmVersionParseError::ComponentMustNotHaveLeadingZeros)
117 } else if part.starts_with('-') || part.starts_with('+') {
118 Err(LlvmVersionParseError::ComponentMustNotHaveSign)
119 } else {
120 Ok(part.parse()?)
121 }
122 });
123
124 let major = parts.next().unwrap()?;
125 let mut minor = 0;
126
127 if let Some(part) = parts.next() {
128 minor = part?;
129 } else if major < 4 {
130 return Err(LlvmVersionParseError::MinorVersionRequiredBefore4);
132 }
133
134 if let Some(Err(e)) = parts.next() {
135 return Err(e);
136 }
137
138 if parts.next().is_some() {
139 return Err(LlvmVersionParseError::TooManyComponents);
140 }
141
142 Ok(Self { major, minor })
143 }
144}
145
146#[derive(Clone, Debug, Eq, PartialEq, Ord, PartialOrd, Hash)]
148pub struct VersionMeta {
149 pub semver: Version,
151
152 pub commit_hash: Option<String>,
154
155 pub commit_date: Option<String>,
157
158 pub build_date: Option<String>,
160
161 pub channel: Channel,
163
164 pub host: String,
166
167 pub short_version_string: String,
169
170 pub llvm_version: Option<LlvmVersion>,
172}
173
174impl VersionMeta {
175 pub fn for_command(mut cmd: Command) -> Result<VersionMeta> {
177 let out = cmd
178 .arg("-vV")
179 .output()
180 .map_err(Error::CouldNotExecuteCommand)?;
181
182 if !out.status.success() {
183 return Err(Error::CommandError {
184 stdout: String::from_utf8_lossy(&out.stdout).into(),
185 stderr: String::from_utf8_lossy(&out.stderr).into(),
186 });
187 }
188
189 version_meta_for(str::from_utf8(&out.stdout)?)
190 }
191}
192
193pub fn version() -> Result<Version> {
195 Ok(version_meta()?.semver)
196}
197
198pub fn version_meta() -> Result<VersionMeta> {
201 let rustc = env::var_os("RUSTC").unwrap_or_else(|| OsString::from("rustc"));
202 let cmd = if let Some(wrapper) = env::var_os("RUSTC_WRAPPER").filter(|w| !w.is_empty()) {
203 let mut cmd = Command::new(wrapper);
204 cmd.arg(rustc);
205 cmd
206 } else {
207 Command::new(rustc)
208 };
209
210 VersionMeta::for_command(cmd)
211}
212
213pub fn version_meta_for(verbose_version_string: &str) -> Result<VersionMeta> {
217 let mut map = HashMap::new();
218 for (i, line) in verbose_version_string.lines().enumerate() {
219 if i == 0 {
220 map.insert("short", line);
221 continue;
222 }
223
224 let mut parts = line.splitn(2, ": ");
225 let key = match parts.next() {
226 Some(key) => key,
227 None => continue,
228 };
229
230 if let Some(value) = parts.next() {
231 map.insert(key, value);
232 }
233 }
234
235 let short_version_string = expect_key("short", &map)?;
236 let host = expect_key("host", &map)?;
237 let release = expect_key("release", &map)?;
238 let semver: Version = release.parse()?;
239
240 let channel = match semver.pre.split('.').next().unwrap() {
241 "" => Channel::Stable,
242 "dev" => Channel::Dev,
243 "beta" => Channel::Beta,
244 "nightly" => Channel::Nightly,
245 x => return Err(Error::UnknownPreReleaseTag(x.to_owned())),
246 };
247
248 let commit_hash = expect_key_or_unknown("commit-hash", &map)?;
249 let commit_date = expect_key_or_unknown("commit-date", &map)?;
250 let build_date = map
251 .get("build-date")
252 .filter(|&v| *v != "unknown")
253 .map(|&v| String::from(v));
254 let llvm_version = match map.get("LLVM version") {
255 Some(&v) => Some(v.parse()?),
256 None => None,
257 };
258
259 Ok(VersionMeta {
260 semver,
261 commit_hash,
262 commit_date,
263 build_date,
264 channel,
265 host,
266 short_version_string,
267 llvm_version,
268 })
269}
270
271fn expect_key_or_unknown(key: &str, map: &HashMap<&str, &str>) -> Result<Option<String>, Error> {
272 match map.get(key) {
273 Some(&"unknown") => Ok(None),
274 Some(&v) => Ok(Some(String::from(v))),
275 None => Err(Error::UnexpectedVersionFormat),
276 }
277}
278
279fn expect_key(key: &str, map: &HashMap<&str, &str>) -> Result<String, Error> {
280 map.get(key)
281 .map(|&v| String::from(v))
282 .ok_or(Error::UnexpectedVersionFormat)
283}
284
285#[derive(Debug)]
287pub enum LlvmVersionParseError {
288 ParseIntError(num::ParseIntError),
290 ComponentMustNotHaveLeadingZeros,
292 ComponentMustNotHaveSign,
294 MinorVersionMustBeZeroAfter4,
296 MinorVersionRequiredBefore4,
298 TooManyComponents,
300}
301
302impl From<num::ParseIntError> for LlvmVersionParseError {
303 fn from(e: num::ParseIntError) -> Self {
304 LlvmVersionParseError::ParseIntError(e)
305 }
306}
307
308impl fmt::Display for LlvmVersionParseError {
309 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
310 match self {
311 LlvmVersionParseError::ParseIntError(e) => {
312 write!(f, "error parsing LLVM version component: {}", e)
313 }
314 LlvmVersionParseError::ComponentMustNotHaveLeadingZeros => {
315 write!(f, "a version component must not have leading zeros")
316 }
317 LlvmVersionParseError::ComponentMustNotHaveSign => {
318 write!(f, "a version component must not have a sign")
319 }
320 LlvmVersionParseError::MinorVersionMustBeZeroAfter4 => write!(
321 f,
322 "LLVM's minor version component must be 0 for versions greater than 4.0"
323 ),
324 LlvmVersionParseError::MinorVersionRequiredBefore4 => write!(
325 f,
326 "LLVM's minor version component is required for versions less than 4.0"
327 ),
328 LlvmVersionParseError::TooManyComponents => write!(f, "too many version components"),
329 }
330 }
331}
332
333impl error::Error for LlvmVersionParseError {
334 fn source(&self) -> Option<&(dyn error::Error + 'static)> {
335 match self {
336 LlvmVersionParseError::ParseIntError(e) => Some(e),
337 LlvmVersionParseError::ComponentMustNotHaveLeadingZeros
338 | LlvmVersionParseError::ComponentMustNotHaveSign
339 | LlvmVersionParseError::MinorVersionMustBeZeroAfter4
340 | LlvmVersionParseError::MinorVersionRequiredBefore4
341 | LlvmVersionParseError::TooManyComponents => None,
342 }
343 }
344}
345
346#[derive(Debug)]
348pub enum Error {
349 CouldNotExecuteCommand(io::Error),
351 CommandError {
353 stdout: String,
355 stderr: String,
357 },
358 Utf8Error(str::Utf8Error),
360 UnexpectedVersionFormat,
362 SemVerError(semver::Error),
364 UnknownPreReleaseTag(String),
366 LlvmVersionError(LlvmVersionParseError),
368}
369
370impl fmt::Display for Error {
371 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
372 match *self {
373 CouldNotExecuteCommand(ref e) => write!(f, "could not execute command: {}", e),
374 CommandError {
375 ref stdout,
376 ref stderr,
377 } => write!(
378 f,
379 "error from command -- stderr:\n\n{}\n\nstderr:\n\n{}",
380 stderr, stdout,
381 ),
382 Utf8Error(_) => write!(f, "invalid UTF-8 output from `rustc -vV`"),
383 UnexpectedVersionFormat => write!(f, "unexpected `rustc -vV` format"),
384 SemVerError(ref e) => write!(f, "error parsing version: {}", e),
385 UnknownPreReleaseTag(ref i) => write!(f, "unknown pre-release tag: {}", i),
386 LlvmVersionError(ref e) => write!(f, "error parsing LLVM's version: {}", e),
387 }
388 }
389}
390
391impl error::Error for Error {
392 fn source(&self) -> Option<&(dyn error::Error + 'static)> {
393 match *self {
394 CouldNotExecuteCommand(ref e) => Some(e),
395 CommandError { .. } => None,
396 Utf8Error(ref e) => Some(e),
397 UnexpectedVersionFormat => None,
398 SemVerError(ref e) => Some(e),
399 UnknownPreReleaseTag(_) => None,
400 LlvmVersionError(ref e) => Some(e),
401 }
402 }
403}
404
405macro_rules! impl_from {
406 ($($err_ty:ty => $variant:ident),* $(,)*) => {
407 $(
408 impl From<$err_ty> for Error {
409 fn from(e: $err_ty) -> Error {
410 Error::$variant(e)
411 }
412 }
413 )*
414 }
415}
416
417impl_from! {
418 str::Utf8Error => Utf8Error,
419 semver::Error => SemVerError,
420 LlvmVersionParseError => LlvmVersionError,
421}
422
423pub type Result<T, E = Error> = std::result::Result<T, E>;