Support for `--h_out` cmdline parameter.
PiperOrigin-RevId: 475880360
diff --git a/cc_bindings_from_rs/cmdline.rs b/cc_bindings_from_rs/cmdline.rs
new file mode 100644
index 0000000..f93d4a0
--- /dev/null
+++ b/cc_bindings_from_rs/cmdline.rs
@@ -0,0 +1,116 @@
+// Part of the Crubit project, under the Apache License v2.0 with LLVM
+// Exceptions. See /LICENSE for license information.
+// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
+
+use getopts::{Fail, Options};
+use std::path::Path;
+
+#[derive(Debug)]
+pub struct Cmdline {
+ h_out: String,
+ rustc_args: Vec<String>,
+}
+
+const H_OUT: &str = "h_out";
+
+impl Cmdline {
+ pub fn new(args: &[String]) -> Result<Self, Fail> {
+ // Ensure that `@file` expansion also covers *our* args.
+ let args = rustc_driver::args::arg_expand_all(args);
+
+ let matches = Options::new()
+ .reqopt("", H_OUT, "output path for C++ header file with bindings", "FILE")
+ .parse(&args)?;
+ let h_out =
+ matches.opt_str(H_OUT).expect("getopts should enforce presence of --h_out `reqopt`");
+
+ Ok(Self { h_out, rustc_args: matches.free })
+ }
+
+ pub fn h_out(&self) -> &Path {
+ Path::new(&self.h_out)
+ }
+
+ pub fn rustc_args(&self) -> &[String] {
+ &self.rustc_args
+ }
+}
+
+#[cfg(test)]
+mod tests {
+ use super::*;
+
+ use itertools::Itertools;
+ use std::io::Write;
+ use tempfile::tempdir;
+
+ fn new_cmdline(args: &[&str]) -> Result<Cmdline, Fail> {
+ let args = args.iter().map(|s| s.to_string()).collect_vec();
+ Cmdline::new(&args)
+ }
+
+ #[test]
+ fn test_h_out_happy_path() -> Result<(), Fail> {
+ let cmdline = new_cmdline(&["--h_out=foo.h"])?;
+ assert_eq!(Path::new("foo.h"), cmdline.h_out());
+ Ok(())
+ }
+
+ #[test]
+ fn test_h_out_missing() {
+ match new_cmdline(&[]) {
+ Err(Fail::OptionMissing(arg)) if arg == H_OUT => (),
+ other => panic!("Unexpected success or unrecognized error: {:?}", other),
+ }
+ }
+
+ #[test]
+ fn test_h_out_without_arg() {
+ match new_cmdline(&["--h_out"]) {
+ Err(Fail::ArgumentMissing(arg)) if arg == H_OUT => (),
+ other => panic!("Unexpected success or unrecognized error: {:?}", other),
+ }
+ }
+
+ #[test]
+ fn test_h_out_duplicated() {
+ match new_cmdline(&["--h_out=foo.h", "--h_out=bar.h"]) {
+ Err(Fail::OptionDuplicated(arg)) if arg == H_OUT => (),
+ other => panic!("Unexpected success or unrecognized error: {:?}", other),
+ }
+ }
+
+ #[test]
+ fn test_rustc_args_happy_path() -> Result<(), Fail> {
+ // Note that this test would fail without the `--` separator.
+ let cmdline = new_cmdline(&["--h_out=foo.h", "--", "test.rs", "--crate-type=lib"])?;
+ let rustc_args = cmdline.rustc_args();
+ assert!(
+ itertools::equal(&["test.rs", "--crate-type=lib"], rustc_args),
+ "rustc_args = {:?}",
+ rustc_args
+ );
+ Ok(())
+ }
+
+ #[test]
+ fn test_here_file() -> anyhow::Result<()> {
+ let tmpdir = tempdir()?;
+ let tmpfile = tmpdir.path().join("herefile");
+ let mut f = std::fs::File::create(tmpfile.clone())?;
+ writeln!(f, "--h_out=foo.h")?;
+ writeln!(f, "--")?;
+ writeln!(f, "test.rs")?;
+ writeln!(f, "--crate-type=lib")?;
+
+ let cmdline = Cmdline::new(&[format!("@{}", tmpfile.display())])?;
+ assert_eq!(Path::new("foo.h"), cmdline.h_out());
+ let rustc_args = cmdline.rustc_args();
+ assert!(
+ itertools::equal(&["test.rs", "--crate-type=lib"], rustc_args),
+ "rustc_args = {:?}",
+ rustc_args
+ );
+ Ok(())
+ }
+}