Browse Source

Merge pull request #128 from sfackler/master

Add a method to macro-expand source files
vs2017
Alex Crichton 8 years ago
committed by GitHub
parent
commit
9b65ef4c3f
  1. 2
      .gitignore
  2. 6
      gcc-test/build.rs
  3. 4
      gcc-test/src/expand.c
  4. 59
      src/lib.rs

2
.gitignore

@ -1,2 +1,4 @@
target target
Cargo.lock Cargo.lock
.idea
*.iml

6
gcc-test/build.rs

@ -81,4 +81,10 @@ fn main() {
.cargo_metadata(false) .cargo_metadata(false)
.file("src/opt_linkage.c") .file("src/opt_linkage.c")
.compile("libOptLinkage.a"); .compile("libOptLinkage.a");
let out = gcc::Config::new()
.file("src/expand.c")
.expand();
let out = String::from_utf8(out).unwrap();
assert!(out.contains("hello world"));
} }

4
gcc-test/src/expand.c

@ -0,0 +1,4 @@
#define HELLO hello
#define WORLD world
HELLO WORLD

59
src/lib.rs

@ -52,10 +52,10 @@ extern crate rayon;
use std::env; use std::env;
use std::ffi::{OsString, OsStr}; use std::ffi::{OsString, OsStr};
use std::fs; use std::fs;
use std::io;
use std::path::{PathBuf, Path}; use std::path::{PathBuf, Path};
use std::process::{Command, Stdio}; use std::process::{Command, Stdio};
use std::io::{BufReader, BufRead, Write}; use std::io::{self, BufReader, BufRead, Read, Write};
use std::thread;
#[cfg(windows)] #[cfg(windows)]
mod registry; mod registry;
@ -131,6 +131,15 @@ impl ToolFamily {
ToolFamily::Clang => "-I", ToolFamily::Clang => "-I",
} }
} }
/// What the flag to request macro-expanded source output looks like
fn expand_flag(&self) -> &'static str {
match *self {
ToolFamily::Msvc => "/E",
ToolFamily::Gnu |
ToolFamily::Clang => "-E",
}
}
} }
/// Compile a library from the given set of input C files. /// Compile a library from the given set of input C files.
@ -473,6 +482,29 @@ impl Config {
run(&mut cmd, &name); run(&mut cmd, &name);
} }
/// Run the compiler, returning the macro-expanded version of the input files.
///
/// This is only relevant for C and C++ files.
pub fn expand(&self) -> Vec<u8> {
let compiler = self.get_compiler();
let mut cmd = compiler.to_command();
for &(ref a, ref b) in self.env.iter() {
cmd.env(a, b);
}
cmd.arg(compiler.family.expand_flag());
for file in self.files.iter() {
cmd.arg(file);
}
let name = compiler.path
.file_name()
.unwrap()
.to_string_lossy()
.into_owned();
run(&mut cmd, &name)
}
/// Get the compiler that's in use for this configuration. /// Get the compiler that's in use for this configuration.
/// ///
/// This function will return a `Tool` which represents the culmination /// This function will return a `Tool` which represents the culmination
@ -1044,23 +1076,27 @@ impl Tool {
} }
} }
fn run(cmd: &mut Command, program: &str) { fn run(cmd: &mut Command, program: &str) -> Vec<u8> {
println!("running: {:?}", cmd); println!("running: {:?}", cmd);
// Capture the standard error coming from these programs, and write it out // Capture the standard error coming from these programs, and write it out
// with cargo:warning= prefixes. Note that this is a bit wonky to avoid // with cargo:warning= prefixes. Note that this is a bit wonky to avoid
// requiring the output to be UTF-8, we instead just ship bytes from one // requiring the output to be UTF-8, we instead just ship bytes from one
// location to another. // location to another.
let spawn_result = match cmd.stderr(Stdio::piped()).spawn() { let (spawn_result, stdout) = match cmd.stdout(Stdio::piped()).stderr(Stdio::piped()).spawn() {
Ok(mut child) => { Ok(mut child) => {
let stderr = BufReader::new(child.stderr.take().unwrap()); let stderr = BufReader::new(child.stderr.take().unwrap());
for line in stderr.split(b'\n').filter_map(|l| l.ok()) { thread::spawn(move || {
print!("cargo:warning="); for line in stderr.split(b'\n').filter_map(|l| l.ok()) {
std::io::stdout().write_all(&line).unwrap(); print!("cargo:warning=");
println!(""); std::io::stdout().write_all(&line).unwrap();
} println!("");
child.wait() }
});
let mut stdout = vec![];
child.stdout.take().unwrap().read_to_end(&mut stdout).unwrap();
(child.wait(), stdout)
} }
Err(e) => Err(e), Err(e) => (Err(e), vec![]),
}; };
let status = match spawn_result { let status = match spawn_result {
Ok(status) => status, Ok(status) => status,
@ -1083,6 +1119,7 @@ fn run(cmd: &mut Command, program: &str) {
if !status.success() { if !status.success() {
fail(&format!("command did not execute successfully, got: {}", status)); fail(&format!("command did not execute successfully, got: {}", status));
} }
stdout
} }
fn fail(s: &str) -> ! { fn fail(s: &str) -> ! {

Loading…
Cancel
Save