#include "rr.h" #include "v8_handle.h" using namespace v8; v8_handle::v8_handle(Handle handle) : handle(Persistent::New(handle)) { this->weakref_callback = Qnil; this->weakref_callback_parameters = Qnil; this->dead = false; } v8_handle::~v8_handle() {} namespace { VALUE handle_queue; void v8_handle_mark(v8_handle* handle) { rb_gc_mark(handle->weakref_callback); rb_gc_mark(handle->weakref_callback_parameters); } void v8_handle_free(v8_handle* handle) { delete handle; } void v8_handle_enqueue(v8_handle* handle) { handle->dead = true; VALUE zombie = Data_Wrap_Struct(rr_v8_handle_class(), 0, v8_handle_free, handle); rb_ary_unshift(handle_queue, zombie); } void v8_handle_dequeue(GCType type, GCCallbackFlags flags) { for (VALUE handle = rb_ary_pop(handle_queue); RTEST(handle); handle = rb_ary_pop(handle_queue)) { v8_handle* dead = NULL; Data_Get_Struct(handle, struct v8_handle, dead); dead->handle.Dispose(); dead->handle.Clear(); } } VALUE New(VALUE self, VALUE handle) { if (RTEST(handle)) { Persistent that = rr_v8_handle(handle); return rr_v8_handle_new(self, that); } else { return rr_v8_handle_new(self, Handle()); } } VALUE IsEmpty(VALUE self) { return rr_v82rb(rr_v8_handle(self).IsEmpty()); } VALUE Clear(VALUE self) { rr_v8_handle(self).Clear(); return Qnil; } VALUE Dispose(VALUE self) { rr_v8_handle(self).Dispose(); return Qnil; } void RubyWeakReferenceCallback(Persistent value, void* parameter) { VALUE self = (VALUE)parameter; v8_handle* handle = rr_v8_handle_raw(self); VALUE callback = handle->weakref_callback; VALUE parameters = handle->weakref_callback_parameters; if (RTEST(callback)) { rb_funcall(callback, rb_intern("call"), 2, self, parameters); } value.Dispose(); handle->handle.Dispose(); handle->handle.Clear(); handle->dead = true; } VALUE MakeWeak(VALUE self, VALUE parameters, VALUE callback) { v8_handle* handle = rr_v8_handle_raw(self); handle->weakref_callback = callback; handle->weakref_callback_parameters = parameters; rr_v8_handle(self).MakeWeak((void*)self, RubyWeakReferenceCallback); return Qnil; } VALUE ClearWeak(VALUE self) { rr_v8_handle(self).ClearWeak(); return Qnil; } VALUE IsNearDeath(VALUE self) { return rr_v82rb(rr_v8_handle(self).IsNearDeath()); } VALUE IsWeak(VALUE self) { return rr_v82rb(rr_v8_handle(self).IsWeak()); } VALUE dead_p(VALUE self) { return rr_v8_handle_raw(self)->dead ? Qtrue : Qfalse; } } void rr_init_handle() { VALUE HandleClass = rr_define_class("Handle"); rr_define_method(HandleClass, "dead?", dead_p, 0); rr_define_singleton_method(HandleClass, "New", New, 1); rr_define_method(HandleClass, "IsEmpty", IsEmpty, 0); rr_define_method(HandleClass, "Clear", Clear, 0); rr_define_method(HandleClass, "Dispose", Dispose, 0); rr_define_method(HandleClass, "MakeWeak", MakeWeak, 2); rr_define_method(HandleClass, "ClearWeak", ClearWeak, 0); rr_define_method(HandleClass, "IsNearDeath", IsNearDeath, 0); rr_define_method(HandleClass, "IsWeak", IsWeak, 0); handle_queue = rb_ary_new(); rb_gc_register_address(&handle_queue); V8::AddGCPrologueCallback(v8_handle_dequeue); } VALUE rr_v8_handle_new(VALUE klass, v8::Handle handle) { v8_handle* new_handle = new v8_handle(handle); return Data_Wrap_Struct(klass, v8_handle_mark, v8_handle_enqueue, new_handle); } VALUE rr_v8_handle_class() { return rr_define_class("Handle"); } v8_handle* rr_v8_handle_raw(VALUE value) { v8_handle* handle = 0; Data_Get_Struct(value, struct v8_handle, handle); return handle; }