0
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-30 11:15:13 -05:00
denoland-deno/src/main.rs

110 lines
3.1 KiB
Rust
Raw Normal View History

2018-06-16 01:43:23 +02:00
extern crate libc;
2018-07-16 23:25:50 -04:00
#[macro_use]
extern crate log;
2018-07-04 14:50:28 -04:00
2018-06-16 01:43:23 +02:00
use libc::c_int;
use std::ffi::CStr;
use std::ffi::CString;
2018-06-22 23:19:43 +02:00
use std::ptr;
mod binding;
2018-07-19 21:06:48 -04:00
use binding::{
deno_delete, deno_execute, deno_handle_msg_from_js, deno_init,
deno_last_exception, deno_new, deno_set_flags, DenoC,
};
2018-06-16 01:43:23 +02:00
// 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.
2018-06-16 01:43:23 +02:00
let mut argv = std::env::args()
.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)
2018-06-16 01:43:23 +02:00
.collect::<Vec<_>>();
// 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.
2018-06-16 01:43:23 +02:00
unsafe {
deno_set_flags(&mut c_argc, c_argv.as_mut_ptr());
2018-06-16 01:43:23 +02:00
};
// 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<_>>()
2018-06-16 01:43:23 +02:00
}
2018-06-29 15:07:19 -07:00
type DenoException<'a> = &'a str;
2018-06-22 23:19:43 +02:00
struct Deno {
ptr: *const DenoC,
}
impl Deno {
fn new() -> Deno {
let ptr = unsafe { deno_new(ptr::null(), deno_handle_msg_from_js) };
2018-06-22 23:19:43 +02:00
Deno { ptr: ptr }
}
2018-07-19 21:06:48 -04:00
fn execute(
&mut self,
js_filename: &str,
js_source: &str,
) -> Result<(), DenoException> {
2018-06-22 23:19:43 +02:00
let filename = CString::new(js_filename).unwrap();
let source = CString::new(js_source).unwrap();
2018-07-19 21:06:48 -04:00
let r = unsafe {
deno_execute(self.ptr, filename.as_ptr(), source.as_ptr())
};
2018-06-29 15:07:19 -07:00
if r == 0 {
let ptr = unsafe { deno_last_exception(self.ptr) };
let cstr = unsafe { CStr::from_ptr(ptr) };
return Err(cstr.to_str().unwrap());
}
Ok(())
2018-06-22 23:19:43 +02:00
}
2018-06-29 15:07:19 -07:00
}
2018-06-22 23:19:43 +02:00
2018-06-29 15:07:19 -07:00
impl Drop for Deno {
fn drop(&mut self) {
2018-06-22 23:19:43 +02:00
unsafe { deno_delete(self.ptr) }
}
}
2018-06-16 01:43:23 +02:00
fn main() {
2018-07-16 23:25:50 -04:00
log::set_max_level(log::LevelFilter::Debug);
2018-06-16 01:43:23 +02:00
unsafe { deno_init() };
2018-06-22 23:19:43 +02:00
set_flags();
/*
2018-06-16 01:43:23 +02:00
let v = unsafe { deno_v8_version() };
let c_str = unsafe { CStr::from_ptr(v) };
let version = c_str.to_str().unwrap();
println!("version: {}", version);
2018-06-22 23:19:43 +02:00
*/
2018-06-29 15:07:19 -07:00
let mut d = Deno::new();
2018-06-22 23:19:43 +02:00
2018-07-12 18:38:00 -04:00
d.execute("deno_main.js", "denoMain();")
.unwrap_or_else(|err| {
2018-07-16 23:25:50 -04:00
error!("{}", err);
std::process::exit(1);
2018-07-12 18:38:00 -04:00
});
2018-06-16 01:43:23 +02:00
}