|
|
@ -21,7 +21,7 @@ |
|
|
|
*/ |
|
|
|
|
|
|
|
#include <memory> |
|
|
|
#include <libplatform/libplatform.h> |
|
|
|
//#include <libplatform/libplatform.h>
|
|
|
|
#include "JSV8Engine.h" |
|
|
|
#include "libjsengine/JSEngineResources.hpp" |
|
|
|
|
|
|
@ -42,59 +42,59 @@ static const char* toCString(v8::String::Utf8Value const& value) |
|
|
|
// from https://github.com/v8/v8-git-mirror/blob/master/samples/shell.cc
|
|
|
|
static void reportException(v8::Isolate* isolate, v8::TryCatch* try_catch) |
|
|
|
{ |
|
|
|
v8::HandleScope handle_scope(isolate); |
|
|
|
v8::String::Utf8Value exception(try_catch->Exception()); |
|
|
|
const char* exception_string = toCString(exception); |
|
|
|
v8::Handle<v8::Message> message = try_catch->Message(); |
|
|
|
|
|
|
|
// v8::HandleScope handle_scope;
|
|
|
|
// v8::String::Utf8Value exception(try_catch->Exception());
|
|
|
|
// const char* exception_string = toCString(exception);
|
|
|
|
// v8::Handle<v8::Message> message = try_catch->Message();
|
|
|
|
//
|
|
|
|
// V8 didn't provide any extra information about this error; just
|
|
|
|
// print the exception.
|
|
|
|
if (message.IsEmpty()) |
|
|
|
fprintf(stderr, "%s\n", exception_string); |
|
|
|
else |
|
|
|
{ |
|
|
|
// if (message.IsEmpty())
|
|
|
|
// fprintf(stderr, "%s\n", exception_string);
|
|
|
|
// else
|
|
|
|
// {
|
|
|
|
// Print (filename):(line number): (message).
|
|
|
|
v8::String::Utf8Value filename(message->GetScriptOrigin().ResourceName()); |
|
|
|
const char* filename_string = toCString(filename); |
|
|
|
int linenum = message->GetLineNumber(); |
|
|
|
fprintf(stderr, "%s:%i: %s\n", filename_string, linenum, exception_string); |
|
|
|
|
|
|
|
// v8::String::Utf8Value filename(message->GetScriptOrigin().ResourceName());
|
|
|
|
// const char* filename_string = toCString(filename);
|
|
|
|
// int linenum = message->GetLineNumber();
|
|
|
|
// fprintf(stderr, "%s:%i: %s\n", filename_string, linenum, exception_string);
|
|
|
|
//
|
|
|
|
// Print line of source code.
|
|
|
|
v8::String::Utf8Value sourceline(message->GetSourceLine()); |
|
|
|
const char* sourceline_string = toCString(sourceline); |
|
|
|
fprintf(stderr, "%s\n", sourceline_string); |
|
|
|
|
|
|
|
// v8::String::Utf8Value sourceline(message->GetSourceLine());
|
|
|
|
// const char* sourceline_string = toCString(sourceline);
|
|
|
|
// fprintf(stderr, "%s\n", sourceline_string);
|
|
|
|
//
|
|
|
|
// Print wavy underline (GetUnderline is deprecated).
|
|
|
|
int start = message->GetStartColumn(); |
|
|
|
|
|
|
|
for (int i = 0; i < start; i++) |
|
|
|
fprintf(stderr, " "); |
|
|
|
|
|
|
|
int end = message->GetEndColumn(); |
|
|
|
|
|
|
|
for (int i = start; i < end; i++) |
|
|
|
fprintf(stderr, "^"); |
|
|
|
|
|
|
|
fprintf(stderr, "\n"); |
|
|
|
v8::String::Utf8Value stack_trace(try_catch->StackTrace()); |
|
|
|
|
|
|
|
if (stack_trace.length() > 0) |
|
|
|
{ |
|
|
|
const char* stack_trace_string = toCString(stack_trace); |
|
|
|
fprintf(stderr, "%s\n", stack_trace_string); |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
class ShellArrayBufferAllocator : public v8::ArrayBuffer::Allocator { |
|
|
|
public: |
|
|
|
virtual void* Allocate(size_t length) { |
|
|
|
void* data = AllocateUninitialized(length); |
|
|
|
return data == NULL ? data : memset(data, 0, length); |
|
|
|
} |
|
|
|
virtual void* AllocateUninitialized(size_t length) { return malloc(length); } |
|
|
|
virtual void Free(void* data, size_t) { free(data); } |
|
|
|
}; |
|
|
|
// int start = message->GetStartColumn();
|
|
|
|
//
|
|
|
|
// for (int i = 0; i < start; i++)
|
|
|
|
// fprintf(stderr, " ");
|
|
|
|
//
|
|
|
|
// int end = message->GetEndColumn();
|
|
|
|
//
|
|
|
|
// for (int i = start; i < end; i++)
|
|
|
|
// fprintf(stderr, "^");
|
|
|
|
//
|
|
|
|
// fprintf(stderr, "\n");
|
|
|
|
// v8::String::Utf8Value stack_trace(try_catch->StackTrace());
|
|
|
|
//
|
|
|
|
// if (stack_trace.length() > 0)
|
|
|
|
// {
|
|
|
|
// const char* stack_trace_string = toCString(stack_trace);
|
|
|
|
// fprintf(stderr, "%s\n", stack_trace_string);
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
} |
|
|
|
|
|
|
|
//class ShellArrayBufferAllocator : public v8::ArrayBuffer::Allocator {
|
|
|
|
//public:
|
|
|
|
// virtual void* Allocate(size_t length) {
|
|
|
|
// void* data = AllocateUninitialized(length);
|
|
|
|
// return data == NULL ? data : memset(data, 0, length);
|
|
|
|
// }
|
|
|
|
// virtual void* AllocateUninitialized(size_t length) { return malloc(length); }
|
|
|
|
// virtual void Free(void* data, size_t) { free(data); }
|
|
|
|
//};
|
|
|
|
|
|
|
|
class JSV8Env |
|
|
|
{ |
|
|
@ -104,32 +104,37 @@ public: |
|
|
|
~JSV8Env(); |
|
|
|
|
|
|
|
private: |
|
|
|
v8::Platform *m_platform; |
|
|
|
// v8::Platform *m_platform;
|
|
|
|
}; |
|
|
|
|
|
|
|
v8::Handle<v8::Context> createShellContext(v8::Isolate* isolate) |
|
|
|
v8::Handle<v8::Context> createShellContext() |
|
|
|
{ |
|
|
|
v8::Handle<v8::ObjectTemplate> global = v8::ObjectTemplate::New(isolate); |
|
|
|
v8::Handle<v8::Context> context = v8::Context::New(isolate, NULL, global); |
|
|
|
return context; |
|
|
|
v8::Handle<v8::ObjectTemplate> global = v8::ObjectTemplate::New(); |
|
|
|
return v8::Context::New(NULL, global); |
|
|
|
} |
|
|
|
|
|
|
|
class JSV8Scope |
|
|
|
{ |
|
|
|
public: |
|
|
|
JSV8Scope(v8::Isolate* _isolate): |
|
|
|
m_isolateScope(_isolate), |
|
|
|
m_handleScope(_isolate), |
|
|
|
m_context(createShellContext(_isolate)), |
|
|
|
JSV8Scope(): |
|
|
|
m_handleScope(), |
|
|
|
m_context(createShellContext()), |
|
|
|
m_contextScope(m_context) |
|
|
|
{} |
|
|
|
{ |
|
|
|
m_context->Enter(); |
|
|
|
} |
|
|
|
|
|
|
|
~JSV8Scope() |
|
|
|
{ |
|
|
|
m_context->Exit(); |
|
|
|
m_context.Dispose(); |
|
|
|
} |
|
|
|
|
|
|
|
v8::Handle <v8::Context> const& context() const { return m_context; } |
|
|
|
v8::Persistent <v8::Context> const& context() const { return m_context; } |
|
|
|
|
|
|
|
private: |
|
|
|
v8::Isolate::Scope m_isolateScope; |
|
|
|
v8::HandleScope m_handleScope; |
|
|
|
v8::Handle <v8::Context> m_context; |
|
|
|
v8::Persistent <v8::Context> m_context; |
|
|
|
v8::Context::Scope m_contextScope; |
|
|
|
}; |
|
|
|
|
|
|
@ -152,28 +157,22 @@ const char* JSV8Value::asCString() const |
|
|
|
|
|
|
|
JSV8Env::JSV8Env() |
|
|
|
{ |
|
|
|
static bool initialized = false; |
|
|
|
if (initialized) |
|
|
|
return; |
|
|
|
initialized = true; |
|
|
|
v8::V8::InitializeICU(); |
|
|
|
m_platform = v8::platform::CreateDefaultPlatform(); |
|
|
|
v8::V8::InitializePlatform(m_platform); |
|
|
|
v8::V8::Initialize(); |
|
|
|
ShellArrayBufferAllocator array_buffer_allocator; |
|
|
|
v8::V8::SetArrayBufferAllocator(&array_buffer_allocator); |
|
|
|
// v8::V8::InitializeICU();
|
|
|
|
// m_platform = v8::platform::CreateDefaultPlatform();
|
|
|
|
// v8::V8::InitializePlatform(m_platform);
|
|
|
|
// v8::V8::Initialize();
|
|
|
|
// ShellArrayBufferAllocator array_buffer_allocator;
|
|
|
|
// v8::V8::SetArrayBufferAllocator(&array_buffer_allocator);
|
|
|
|
} |
|
|
|
|
|
|
|
JSV8Env::~JSV8Env() |
|
|
|
{ |
|
|
|
v8::V8::Dispose(); |
|
|
|
v8::V8::ShutdownPlatform(); |
|
|
|
delete m_platform; |
|
|
|
// v8::V8::ShutdownPlatform();
|
|
|
|
// delete m_platform;
|
|
|
|
} |
|
|
|
|
|
|
|
JSV8Engine::JSV8Engine(): |
|
|
|
m_isolate(v8::Isolate::New()), |
|
|
|
m_scope(new JSV8Scope(m_isolate)) |
|
|
|
JSV8Engine::JSV8Engine(): m_scope(new JSV8Scope()) |
|
|
|
{ |
|
|
|
JSEngineResources resources; |
|
|
|
string common = resources.loadResourceAsString("common"); |
|
|
@ -186,15 +185,14 @@ JSV8Engine::JSV8Engine(): |
|
|
|
JSV8Engine::~JSV8Engine() |
|
|
|
{ |
|
|
|
delete m_scope; |
|
|
|
m_isolate->Dispose(); |
|
|
|
} |
|
|
|
|
|
|
|
JSV8Value JSV8Engine::eval(const char* _cstr) const |
|
|
|
{ |
|
|
|
v8::HandleScope handleScope(m_isolate); |
|
|
|
v8::HandleScope handleScope; |
|
|
|
v8::TryCatch tryCatch; |
|
|
|
v8::Local<v8::String> source = v8::String::NewFromUtf8(context()->GetIsolate(), _cstr); |
|
|
|
v8::Local<v8::String> name(v8::String::NewFromUtf8(context()->GetIsolate(), "(shell)")); |
|
|
|
v8::Local<v8::String> source = v8::String::New(_cstr); |
|
|
|
v8::Local<v8::String> name(v8::String::New("(shell)")); |
|
|
|
v8::ScriptOrigin origin(name); |
|
|
|
v8::Handle<v8::Script> script = v8::Script::Compile(source, &origin); |
|
|
|
|
|
|
@ -202,16 +200,16 @@ JSV8Value JSV8Engine::eval(const char* _cstr) const |
|
|
|
// the handle returned from the TryCatch is destroyed
|
|
|
|
if (script.IsEmpty()) |
|
|
|
{ |
|
|
|
reportException(context()->GetIsolate(), &tryCatch); |
|
|
|
return v8::Exception::Error(v8::Local<v8::String>::New(context()->GetIsolate(), tryCatch.Message()->Get())); |
|
|
|
// reportException(&tryCatch);
|
|
|
|
return v8::Exception::Error(v8::Local<v8::String>::New(tryCatch.Message()->Get())); |
|
|
|
} |
|
|
|
|
|
|
|
auto result = script->Run(); |
|
|
|
|
|
|
|
if (result.IsEmpty()) |
|
|
|
{ |
|
|
|
reportException(context()->GetIsolate(), &tryCatch); |
|
|
|
return v8::Exception::Error(v8::Local<v8::String>::New(context()->GetIsolate(), tryCatch.Message()->Get())); |
|
|
|
// reportException(&tryCatch);
|
|
|
|
return v8::Exception::Error(v8::Local<v8::String>::New(tryCatch.Message()->Get())); |
|
|
|
} |
|
|
|
|
|
|
|
return result; |
|
|
|