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_char;
|
|
|
|
use libc::c_int;
|
2018-06-22 23:19:43 +02:00
|
|
|
use libc::c_void;
|
2018-06-16 01:43:23 +02:00
|
|
|
use std::ffi::CStr;
|
|
|
|
use std::ffi::CString;
|
2018-06-22 23:19:43 +02:00
|
|
|
use std::ptr;
|
|
|
|
|
|
|
|
#[repr(C)]
|
|
|
|
struct deno_buf {
|
2018-07-09 03:35:34 +02:00
|
|
|
alloc_ptr: *mut u8,
|
|
|
|
alloc_len: usize,
|
|
|
|
data_ptr: *mut u8,
|
|
|
|
data_len: usize,
|
2018-06-22 23:19:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[repr(C)]
|
|
|
|
struct DenoC {
|
|
|
|
_unused: [u8; 0],
|
|
|
|
}
|
|
|
|
|
2018-07-13 03:24:07 -04:00
|
|
|
type DenoRecvCb = unsafe extern "C" fn(d: *const DenoC, buf: deno_buf);
|
2018-06-16 01:43:23 +02:00
|
|
|
|
|
|
|
#[link(name = "deno", kind = "static")]
|
|
|
|
extern "C" {
|
|
|
|
fn deno_init();
|
2018-06-22 23:19:43 +02:00
|
|
|
#[allow(dead_code)]
|
|
|
|
fn deno_v8_version() -> *const c_char;
|
2018-06-19 19:04:56 +02:00
|
|
|
fn deno_set_flags(argc: *mut c_int, argv: *mut *mut c_char);
|
2018-06-22 23:19:43 +02:00
|
|
|
|
2018-07-01 18:07:12 +02:00
|
|
|
fn deno_new(data: *const c_void, cb: DenoRecvCb) -> *const DenoC;
|
2018-06-22 23:19:43 +02:00
|
|
|
fn deno_delete(d: *const DenoC);
|
|
|
|
fn deno_last_exception(d: *const DenoC) -> *const c_char;
|
|
|
|
#[allow(dead_code)]
|
|
|
|
fn deno_set_response(d: *const DenoC, buf: deno_buf);
|
|
|
|
fn deno_execute(d: *const DenoC, js_filename: *const c_char, js_source: *const c_char)
|
|
|
|
-> c_int;
|
2018-07-13 03:24:07 -04:00
|
|
|
|
|
|
|
fn deno_handle_msg_from_js(d: *const DenoC, buf: deno_buf);
|
2018-06-16 01:43:23 +02:00
|
|
|
}
|
|
|
|
|
2018-06-19 19:04:56 +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()
|
2018-06-19 19:04:56 +02:00
|
|
|
.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<_>>();
|
2018-06-19 19:04:56 +02:00
|
|
|
// 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 {
|
2018-06-19 19:04:56 +02:00
|
|
|
deno_set_flags(&mut c_argc, c_argv.as_mut_ptr());
|
2018-06-16 01:43:23 +02:00
|
|
|
};
|
2018-06-19 19:04:56 +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 {
|
2018-07-13 03:24:07 -04:00
|
|
|
let ptr = unsafe { deno_new(ptr::null(), deno_handle_msg_from_js) };
|
2018-06-22 23:19:43 +02:00
|
|
|
Deno { ptr: ptr }
|
|
|
|
}
|
|
|
|
|
2018-06-29 15:07:19 -07: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();
|
|
|
|
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);
|
2018-07-01 18:07:12 +02:00
|
|
|
std::process::exit(1);
|
2018-07-12 18:38:00 -04:00
|
|
|
});
|
2018-06-16 01:43:23 +02:00
|
|
|
}
|