summaryrefslogtreecommitdiff
path: root/deno2/main.rs
diff options
context:
space:
mode:
authorBert Belder <bertbelder@gmail.com>2018-06-19 19:04:56 +0200
committerBert Belder <bertbelder@gmail.com>2018-06-19 19:10:35 +0200
commit559453cf6cc88283bcf8fdeccd387458f5c63165 (patch)
treee16f7a51159872d4d7743dd749a0e9c6c0c6e820 /deno2/main.rs
parentb2b4054f97f0190cfd65efb3923b876f7f821da7 (diff)
Make set_flags() return the remaining non-v8 arguments
Diffstat (limited to 'deno2/main.rs')
-rw-r--r--deno2/main.rs47
1 files changed, 34 insertions, 13 deletions
diff --git a/deno2/main.rs b/deno2/main.rs
index 3415e3a24..2ba31a16c 100644
--- a/deno2/main.rs
+++ b/deno2/main.rs
@@ -8,32 +8,53 @@ use std::ffi::CString;
extern "C" {
fn deno_v8_version() -> *const c_char;
fn deno_init();
-
- // Note: `deno_set_flags` actually takes `char**` as it's second argument,
- // not `const char**`, so this is technically incorrect. However it doesn't
- // actually modify the contents of the strings, so it's not unsafe.
- // TODO: use the correct function signature.
- fn deno_set_flags(argc: *mut c_int, argv: *mut *const c_char);
+ fn deno_set_flags(argc: *mut c_int, argv: *mut *mut c_char);
}
-fn set_flags() {
- // Create a vector of zero terminated c strings.
+// Pass the command line arguments to v8.
+// Returns a vector of command line arguments that v8 did not understand.
+fn set_flags() -> Vec<String> {
+ // deno_set_flags(int* argc, char** argv) mutates argc and argv to remove
+ // flags that v8 understands.
+ // Convert command line arguments to a vector of C strings.
let mut argv = std::env::args()
- .map(|arg| CString::new(arg).unwrap().as_ptr())
+ .map(|arg| CString::new(arg).unwrap().into_bytes_with_nul())
+ .collect::<Vec<_>>();
+ // Make a new array, that can be modified by V8::SetFlagsFromCommandLine(),
+ // containing mutable raw pointers to the individual command line args.
+ let mut c_argv = argv.iter_mut()
+ .map(|arg| arg.as_mut_ptr() as *mut i8)
.collect::<Vec<_>>();
- let mut argc = argv.len() as c_int;
+ // Store the length of the argv array in a local variable. We'll pass a
+ // pointer to this local variable to deno_set_flags(), which then
+ // updates its value.
+ let mut c_argc = argv.len() as c_int;
+ // Let v8 parse the arguments it recognizes and remove them from c_argv.
unsafe {
- // pass the pointer of the vector's internal buffer to a C function
- deno_set_flags(&mut argc, argv.as_mut_ptr());
+ deno_set_flags(&mut c_argc, c_argv.as_mut_ptr());
};
+ // If c_argc was updated we have to change the length of c_argv to match.
+ c_argv.truncate(c_argc as usize);
+ // Copy the modified arguments list into a proper rust vec and return it.
+ c_argv
+ .iter()
+ .map(|ptr| unsafe {
+ let cstr = CStr::from_ptr(*ptr as *const i8);
+ let slice = cstr.to_str().unwrap();
+ slice.to_string()
+ })
+ .collect::<Vec<_>>()
}
fn main() {
println!("Hi");
- set_flags();
+ let args = set_flags();
unsafe { deno_init() };
let v = unsafe { deno_v8_version() };
let c_str = unsafe { CStr::from_ptr(v) };
let version = c_str.to_str().unwrap();
println!("version: {}", version);
+ for arg in args {
+ println!("arg: {}", arg);
+ }
}