2010-05-22 02:16:00 -04:00
|
|
|
#include "rr.h"
|
2010-06-08 07:33:54 -04:00
|
|
|
#include "v8_external.h"
|
|
|
|
|
2011-04-11 00:19:26 -04:00
|
|
|
#include "v8_handle.h"
|
2010-06-08 07:33:54 -04:00
|
|
|
#include "v8_value.h"
|
2010-05-22 02:16:00 -04:00
|
|
|
using namespace v8;
|
|
|
|
|
|
|
|
namespace {
|
2010-05-26 12:54:56 -04:00
|
|
|
VALUE ExternalClass;
|
|
|
|
VALUE references;
|
2010-06-08 03:38:32 -04:00
|
|
|
|
2011-04-22 10:58:42 -04:00
|
|
|
VALUE New(VALUE self, VALUE value) {
|
2010-05-22 02:16:00 -04:00
|
|
|
HandleScope scope;
|
2011-04-22 10:58:42 -04:00
|
|
|
return rr_v8_handle_new(self, rr_v8_external_create(value));
|
2010-05-22 02:16:00 -04:00
|
|
|
}
|
|
|
|
VALUE Unwrap(VALUE self, VALUE value) {
|
2010-06-08 03:38:32 -04:00
|
|
|
HandleScope scope;
|
2010-05-22 02:16:00 -04:00
|
|
|
if (rb_obj_is_kind_of(value, self)) {
|
2011-04-11 00:19:26 -04:00
|
|
|
return (VALUE)External::Unwrap(rr_v8_handle<External>(self));
|
2010-05-22 02:16:00 -04:00
|
|
|
} else {
|
|
|
|
rb_raise(rb_eArgError, "cannot unwrap %s. It is not a kind of %s", RSTRING_PTR(rb_class_name(rb_class_of(value))), RSTRING_PTR(rb_class_name(self)));
|
|
|
|
return Qnil;
|
|
|
|
}
|
|
|
|
}
|
2010-06-08 12:51:09 -04:00
|
|
|
VALUE _Value(VALUE self) {
|
|
|
|
HandleScope scope;
|
2011-04-11 00:19:26 -04:00
|
|
|
return (VALUE)rr_v8_handle<External>(self)->Value();
|
2010-06-08 12:51:09 -04:00
|
|
|
}
|
2010-05-26 12:54:56 -04:00
|
|
|
void GCWeakReferenceCallback(Persistent<Value> object, void* parameter) {
|
|
|
|
Local<External> external(External::Cast(*object));
|
|
|
|
rb_hash_delete(references, rb_obj_id((VALUE)external->Value()));
|
|
|
|
}
|
2010-05-22 02:16:00 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void rr_init_v8_external() {
|
2011-04-11 14:32:14 -04:00
|
|
|
ExternalClass = rr_define_class("External", rr_v8_value_class());
|
2010-05-26 12:54:56 -04:00
|
|
|
references = rb_hash_new();
|
|
|
|
rb_define_const(ExternalClass, "OBJECTS_REFERENCED_FROM_WITHIN_V8", references);
|
2010-06-08 07:37:28 -04:00
|
|
|
rr_define_singleton_method(ExternalClass, "New", New, 1);
|
2010-05-22 02:16:00 -04:00
|
|
|
rr_define_singleton_method(ExternalClass, "Unwrap", Unwrap, 1);
|
2010-06-08 12:51:09 -04:00
|
|
|
rr_define_method(ExternalClass, "Value", _Value, 0);
|
2010-05-26 12:54:56 -04:00
|
|
|
}
|
|
|
|
|
2010-06-08 05:48:13 -04:00
|
|
|
VALUE rr_reflect_v8_external(Handle<Value> external) {
|
2011-04-11 00:19:26 -04:00
|
|
|
return rr_v8_handle_new(ExternalClass, external);
|
2010-06-08 05:48:13 -04:00
|
|
|
}
|
|
|
|
|
2010-05-26 12:54:56 -04:00
|
|
|
Handle<Value> rr_v8_external_create(VALUE value) {
|
|
|
|
rb_hash_aset(references, rb_obj_id(value), value);
|
2010-06-08 07:33:54 -04:00
|
|
|
Local<Value> external(External::New((void *)value));
|
2010-05-26 12:54:56 -04:00
|
|
|
Persistent<Value> record = Persistent<Value>::New(external);
|
|
|
|
record.MakeWeak(NULL, GCWeakReferenceCallback);
|
|
|
|
return external;
|
2010-05-22 02:16:00 -04:00
|
|
|
}
|