#include "v8_standalone.h" VALUE ruby_call_symbol; VALUE ruby_respond_to_ID; VALUE ruby_proc_class; VALUE ruby_method_class; bool is_callable(VALUE& object) { return Qtrue == rb_funcall(object, ruby_respond_to_ID, 1, ruby_call_symbol); } bool is_function(VALUE& object) { VALUE klass = RDATA(object)->basic.klass; return klass == ruby_proc_class || klass == ruby_method_class; } /** * Debugging aid */ VALUE v8_what_is_this(VALUE self, VALUE object) { switch (TYPE(object)) { case T_NIL: printf("nil\n"); break; case T_OBJECT: printf("ordinary object\n"); if (is_callable(object)) { printf("responds to call!
"); } break; case T_CLASS: printf("class[%s]
", rb_class2name(object)); break; case T_MODULE: printf("module\n"); break; case T_FLOAT: printf("floating point number\n"); break; case T_STRING: printf("string\n"); break; case T_REGEXP: printf("regular expression\n"); break; case T_ARRAY: printf("array\n"); break; case T_FIXNUM: printf("Fixnum(31bit integer)\n"); break; case T_HASH: printf("associative array\n"); break; case T_STRUCT: printf("(Ruby) structure\n"); break; case T_BIGNUM: printf("multi precision integer\n"); break; case T_FILE: printf("IO\n"); break; case T_TRUE: printf("true\n"); break; case T_FALSE: printf("false\n"); break; case T_DATA: printf("data... inspecting\n"); if (is_function(object)) { printf("It's a function!!!
"); } if (is_callable(object)) { printf("Responds to call!
"); } else { printf("Does *NOT* respond to call
"); } v8_what_is_this(Qnil, RDATA(object)->basic.klass); break; case T_SYMBOL: printf("symbol\n"); break; default: printf("I have no idea!!!\n"); rb_raise(rb_eTypeError, "not valid value"); break; } return Qnil; }