2012-01-31 16:52:08 -05:00
|
|
|
#ifndef THE_RUBY_RACER
|
|
|
|
#define THE_RUBY_RACER
|
|
|
|
|
|
|
|
#include <v8.h>
|
|
|
|
#include <ruby.h>
|
|
|
|
|
|
|
|
namespace rr {
|
2012-05-01 14:53:01 -04:00
|
|
|
|
2012-05-02 19:15:11 -04:00
|
|
|
class Value {
|
|
|
|
public:
|
|
|
|
Value(v8::Handle<v8::Value> handle);
|
|
|
|
Value(VALUE value);
|
2012-05-03 03:24:15 -04:00
|
|
|
virtual operator VALUE();
|
|
|
|
virtual operator v8::Handle<v8::Value>();
|
2012-05-02 19:15:11 -04:00
|
|
|
static void Init();
|
|
|
|
protected:
|
|
|
|
v8::Handle<v8::Value> value;
|
|
|
|
VALUE object;
|
|
|
|
};
|
|
|
|
|
2012-05-03 03:24:15 -04:00
|
|
|
class GC {
|
2012-05-02 19:15:11 -04:00
|
|
|
public:
|
2012-05-03 05:00:04 -04:00
|
|
|
class Queue {
|
|
|
|
public:
|
|
|
|
Queue();
|
|
|
|
void Enqueue(void* phantom);
|
|
|
|
void* Dequeue();
|
|
|
|
private:
|
|
|
|
struct Node {
|
|
|
|
Node(void* val ) : value(val), next(NULL) { }
|
|
|
|
void* value;
|
|
|
|
Node* next;
|
|
|
|
};
|
|
|
|
Node* first; // for producer only
|
|
|
|
Node* divider;
|
|
|
|
Node* last;
|
|
|
|
};
|
|
|
|
static void Finalize(void* phantom);
|
|
|
|
static void Drain(v8::GCType type, v8::GCCallbackFlags flags);
|
2012-05-02 19:15:11 -04:00
|
|
|
static void Init();
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A Reference to a V8 managed object
|
|
|
|
*/
|
|
|
|
template <class T> class Ref {
|
|
|
|
public:
|
|
|
|
Ref<T>(VALUE wrapper) {
|
|
|
|
Holder* holder = NULL;
|
|
|
|
Data_Get_Struct(wrapper, class Holder, holder) ;
|
|
|
|
this->holder = holder;
|
|
|
|
}
|
|
|
|
virtual operator VALUE() {
|
|
|
|
return holder->value;
|
|
|
|
}
|
|
|
|
virtual operator v8::Handle<T>() {
|
|
|
|
return holder->handle;
|
|
|
|
}
|
|
|
|
static Ref<T> create(v8::Handle<T> handle, VALUE klass) {
|
|
|
|
return Ref<T>(new Holder(handle, klass));
|
|
|
|
}
|
|
|
|
inline v8::Handle<T> operator->() const { return holder->handle; }
|
2012-05-03 14:01:10 -04:00
|
|
|
v8::Handle<T> GetHandle() {return holder->handle;}
|
2012-05-03 03:24:15 -04:00
|
|
|
|
2012-05-02 19:15:11 -04:00
|
|
|
class Holder {
|
|
|
|
friend class Ref;
|
2012-05-01 14:53:01 -04:00
|
|
|
public:
|
2012-05-02 19:15:11 -04:00
|
|
|
Holder(v8::Handle<T> handle, VALUE klass) {
|
|
|
|
this->handle = v8::Persistent<T>::New(handle);
|
|
|
|
this->value = Data_Wrap_Struct(klass, 0, &Holder::enqueue, this);
|
|
|
|
}
|
|
|
|
virtual ~Holder() {
|
|
|
|
handle.Dispose();
|
|
|
|
}
|
|
|
|
protected:
|
|
|
|
VALUE value;
|
|
|
|
v8::Persistent<T> handle;
|
|
|
|
|
|
|
|
static void enqueue(Holder* holder) {
|
|
|
|
holder->value = Qnil;
|
2012-05-03 05:00:04 -04:00
|
|
|
GC::Finalize(holder);
|
2012-05-02 19:15:11 -04:00
|
|
|
}
|
|
|
|
};
|
|
|
|
Ref(Holder* holder) {
|
|
|
|
this->holder = holder;
|
2012-05-01 14:53:01 -04:00
|
|
|
};
|
2012-05-02 19:15:11 -04:00
|
|
|
Holder* holder;
|
|
|
|
};
|
|
|
|
|
2012-05-03 03:24:15 -04:00
|
|
|
class Phantom : public Ref<void> {
|
|
|
|
public:
|
2012-05-03 05:00:04 -04:00
|
|
|
inline Phantom(void* reference) : Ref<void>((Ref<void>::Holder*)reference) {}
|
|
|
|
inline bool NotNull() {
|
|
|
|
return this->holder != NULL;
|
|
|
|
}
|
2012-05-03 03:24:15 -04:00
|
|
|
inline void destroy() {
|
|
|
|
delete holder;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-05-02 19:15:11 -04:00
|
|
|
class Context : public Ref<v8::Context> {
|
|
|
|
public:
|
|
|
|
static void Init();
|
2012-05-03 18:21:25 -04:00
|
|
|
static VALUE New(VALUE self);
|
|
|
|
static VALUE Enter(VALUE self);
|
|
|
|
static VALUE Exit(VALUE self);
|
|
|
|
|
|
|
|
private:
|
|
|
|
inline Context(VALUE value) : Ref<v8::Context>(value) {}
|
2012-05-02 19:15:11 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
class Script : public Ref<v8::Script> {
|
|
|
|
public:
|
|
|
|
static void Init();
|
2012-05-03 18:21:25 -04:00
|
|
|
static VALUE New(VALUE klass, VALUE source, VALUE filename);
|
|
|
|
static VALUE Run(VALUE self);
|
|
|
|
|
|
|
|
private:
|
|
|
|
inline Script(VALUE value) : Ref<v8::Script>(value) {}
|
2012-05-02 19:15:11 -04:00
|
|
|
};
|
2012-05-01 14:53:01 -04:00
|
|
|
|
2012-05-03 03:24:15 -04:00
|
|
|
class String: public Ref<v8::String> {
|
|
|
|
public:
|
2012-05-03 18:21:25 -04:00
|
|
|
static void Init();
|
|
|
|
static VALUE New(VALUE self, VALUE value);
|
|
|
|
static VALUE Utf8Value(VALUE self);
|
|
|
|
|
2012-05-03 03:24:15 -04:00
|
|
|
inline String(VALUE value) : Ref<v8::String>(value) {}
|
2012-05-03 14:01:10 -04:00
|
|
|
virtual operator v8::Handle<v8::Value>();
|
2012-05-03 03:24:15 -04:00
|
|
|
static VALUE ToRuby(v8::Handle<v8::String> value);
|
2012-05-03 18:21:25 -04:00
|
|
|
private:
|
2012-05-03 14:01:10 -04:00
|
|
|
static VALUE Class;
|
2012-05-03 03:24:15 -04:00
|
|
|
};
|
|
|
|
|
2012-05-03 13:56:41 -04:00
|
|
|
class V8 {
|
|
|
|
public:
|
|
|
|
static void Init();
|
2012-05-03 18:21:25 -04:00
|
|
|
static VALUE IdleNotification(VALUE self);
|
2012-05-03 13:56:41 -04:00
|
|
|
};
|
2012-05-03 03:24:15 -04:00
|
|
|
|
2012-05-03 17:34:51 -04:00
|
|
|
class ClassBuilder {
|
|
|
|
public:
|
|
|
|
ClassBuilder(const char* name, VALUE superclass = rb_cObject);
|
|
|
|
ClassBuilder& defineMethod(const char* name, VALUE (*impl)(VALUE));
|
|
|
|
ClassBuilder& defineMethod(const char* name, VALUE (*impl)(VALUE, VALUE));
|
2012-05-03 17:47:23 -04:00
|
|
|
ClassBuilder& defineMethod(const char* name, VALUE (*impl)(VALUE, VALUE, VALUE));
|
2012-05-03 17:34:51 -04:00
|
|
|
ClassBuilder& defineSingletonMethod(const char* name, VALUE (*impl)(VALUE));
|
|
|
|
ClassBuilder& defineSingletonMethod(const char* name, VALUE (*impl)(VALUE, VALUE));
|
2012-05-03 17:47:23 -04:00
|
|
|
ClassBuilder& defineSingletonMethod(const char* name, VALUE (*impl)(VALUE, VALUE, VALUE));
|
2012-05-03 17:34:51 -04:00
|
|
|
inline operator VALUE() {return this->value;}
|
|
|
|
private:
|
|
|
|
VALUE value;
|
|
|
|
};
|
|
|
|
|
2012-01-31 16:52:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|