diff options
Diffstat (limited to 'src/object.c')
| -rw-r--r-- | src/object.c | 632 |
1 files changed, 632 insertions, 0 deletions
diff --git a/src/object.c b/src/object.c new file mode 100644 index 000000000..c60c2fb7b --- /dev/null +++ b/src/object.c @@ -0,0 +1,632 @@ +#include "mruby.h" +#include <string.h> +#include "mruby/string.h" +#include <stdio.h> +#include "mruby/class.h" +#include "method.h" +#include "mruby/numeric.h" +#include "mdata.h" + +#ifdef INCLUDE_REGEXP + #define mrb_usascii_str_new2 mrb_usascii_str_new_cstr +#else + #define mrb_usascii_str_new2 mrb_str_new_cstr + #define mrb_usascii_str_new mrb_str_new +#endif + +#ifndef FALSE +#define FALSE 0 +#endif + +#ifndef TRUE +#define TRUE 1 +#endif + +int +mrb_obj_eq(mrb_state *mrb, mrb_value v1, mrb_value v2) +{ + if (v1.tt != v2.tt) return 0; + switch (v1.tt) { + case MRB_TT_TRUE: + return 1; + + case MRB_TT_FALSE: + case MRB_TT_FIXNUM: + return (v1.value.i == v2.value.i); + case MRB_TT_SYMBOL: + return (v1.value.sym == v2.value.sym); + + case MRB_TT_FLOAT: + return (v1.value.f == v2.value.f); + + default: + return (v1.value.p == v2.value.p); + } +} + +int +mrb_obj_equal(mrb_state *mrb, mrb_value v1, mrb_value v2) +{ + /* temporary definition */ + return mrb_obj_eq(mrb, v1, v2); +} + +int +mrb_equal(mrb_state *mrb, mrb_value obj1, mrb_value obj2) +{ + mrb_value result; + + if (mrb_obj_eq(mrb, obj1, obj2)) return TRUE; + result = mrb_funcall(mrb, obj1, "==", 1, obj2); + if (mrb_nil_p(result)) { + return FALSE; + } + else { + if (mrb_type(result) == MRB_TT_TRUE) { + return TRUE; + } + else { + return FALSE; + } + } +} + +/* + * Document-class: NilClass + * + * The class of the singleton object <code>nil</code>. + */ + +/* 15.2.4.3.4 */ +/* + * call_seq: + * nil.nil? -> true + * + * Only the object <i>nil</i> responds <code>true</code> to <code>nil?</code>. + */ + +static mrb_value +mrb_true(mrb_state *mrb, mrb_value obj) +{ + return mrb_true_value(); +} + +/* 15.2.4.3.5 */ +/* + * call-seq: + * nil.to_s -> "" + * + * Always returns the empty string. + */ + +static mrb_value +nil_to_s(mrb_state *mrb, mrb_value obj) +{ + return mrb_usascii_str_new(mrb, 0, 0); +} + +/*********************************************************************** + * Document-class: TrueClass + * + * The global value <code>true</code> is the only instance of class + * <code>TrueClass</code> and represents a logically true value in + * boolean expressions. The class provides operators allowing + * <code>true</code> to be used in logical expressions. + */ + +/* 15.2.5.3.1 */ +/* + * call-seq: + * true & obj -> true or false + * + * And---Returns <code>false</code> if <i>obj</i> is + * <code>nil</code> or <code>false</code>, <code>true</code> otherwise. + */ + +static mrb_value +true_and(mrb_state *mrb, mrb_value obj) +{ + mrb_value obj2; + + mrb_get_args(mrb, "o", &obj2); + return mrb_test(obj2)?mrb_true_value():mrb_false_value(); +} + +/* 15.2.5.3.2 */ +/* + * call-seq: + * true ^ obj -> !obj + * + * Exclusive Or---Returns <code>true</code> if <i>obj</i> is + * <code>nil</code> or <code>false</code>, <code>false</code> + * otherwise. + */ + +static mrb_value +true_xor(mrb_state *mrb, mrb_value obj) +{ + mrb_value obj2; + + mrb_get_args(mrb, "o", &obj2); + return mrb_test(obj2)?mrb_false_value():mrb_true_value(); +} + +/* 15.2.5.3.3 */ +/* + * call-seq: + * true.to_s -> "true" + * + * The string representation of <code>true</code> is "true". + */ + +static mrb_value +true_to_s(mrb_state *mrb, mrb_value obj) +{ + return mrb_usascii_str_new2(mrb, "true"); +} + +/* 15.2.5.3.4 */ +/* + * call-seq: + * true | obj -> true + * + * Or---Returns <code>true</code>. As <i>anObject</i> is an argument to + * a method call, it is always evaluated; there is no short-circuit + * evaluation in this case. + * + * true | puts("or") + * true || puts("logical or") + * + * <em>produces:</em> + * + * or + */ + +static mrb_value +true_or(mrb_state *mrb, mrb_value obj) +{ + mrb_value obj2; + + mrb_get_args(mrb, "o", &obj2); + return mrb_true_value(); +} + +/* + * Document-class: FalseClass + * + * The global value <code>false</code> is the only instance of class + * <code>FalseClass</code> and represents a logically false value in + * boolean expressions. The class provides operators allowing + * <code>false</code> to participate correctly in logical expressions. + * + */ + +/* 15.2.4.3.1 */ +/* 15.2.6.3.1 */ +/* + * call-seq: + * false & obj -> false + * nil & obj -> false + * + * And---Returns <code>false</code>. <i>obj</i> is always + * evaluated as it is the argument to a method call---there is no + * short-circuit evaluation in this case. + */ + +static mrb_value +false_and(mrb_state *mrb, mrb_value obj) +{ + mrb_value obj2; + + mrb_get_args(mrb, "o", &obj2); + return mrb_false_value(); +} + +/* 15.2.4.3.2 */ +/* 15.2.6.3.2 */ +/* + * call-seq: + * false ^ obj -> true or false + * nil ^ obj -> true or false + * + * Exclusive Or---If <i>obj</i> is <code>nil</code> or + * <code>false</code>, returns <code>false</code>; otherwise, returns + * <code>true</code>. + * + */ + +static mrb_value +false_xor(mrb_state *mrb, mrb_value obj) +{ + mrb_value obj2; + + mrb_get_args(mrb, "o", &obj2); + return mrb_test(obj2)?mrb_true_value():mrb_false_value(); +} + +/* 15.2.4.3.3 */ +/* 15.2.6.3.4 */ +/* + * call-seq: + * false | obj -> true or false + * nil | obj -> true or false + * + * Or---Returns <code>false</code> if <i>obj</i> is + * <code>nil</code> or <code>false</code>; <code>true</code> otherwise. + */ + +static mrb_value +false_or(mrb_state *mrb, mrb_value obj) +{ + mrb_value obj2; + + mrb_get_args(mrb, "o", &obj2); + return mrb_test(obj2)?mrb_true_value():mrb_false_value(); +} + +/* 15.2.6.3.3 */ +/* + * call-seq: + * false.to_s -> "false" + * + * 'nuf said... + */ + +static mrb_value +false_to_s(mrb_state *mrb, mrb_value obj) +{ + return mrb_usascii_str_new2(mrb, "false"); +} + +void +mrb_init_object(mrb_state *mrb) +{ + struct RClass *n; + struct RClass *t; + struct RClass *f; + + n = mrb->nil_class = mrb_define_class(mrb, "NilClass", mrb->object_class); + mrb_define_method(mrb, n, "&", false_and, ARGS_REQ(1)); /* 15.2.4.3.1 */ + mrb_define_method(mrb, n, "^", false_xor, ARGS_REQ(1)); /* 15.2.4.3.2 */ + mrb_define_method(mrb, n, "|", false_or, ARGS_REQ(1)); /* 15.2.4.3.3 */ + mrb_define_method(mrb, n, "nil?", mrb_true, ARGS_NONE()); /* 15.2.4.3.4 */ + mrb_define_method(mrb, n, "to_s", nil_to_s, ARGS_NONE()); /* 15.2.4.3.5 */ + + t = mrb->true_class = mrb_define_class(mrb, "TrueClass", mrb->object_class); + mrb_define_method(mrb, t, "&", true_and, ARGS_REQ(1)); /* 15.2.5.3.1 */ + mrb_define_method(mrb, t, "^", true_xor, ARGS_REQ(1)); /* 15.2.5.3.2 */ + mrb_define_method(mrb, t, "to_s", true_to_s, ARGS_NONE()); /* 15.2.5.3.3 */ + mrb_define_method(mrb, t, "|", true_or, ARGS_REQ(1)); /* 15.2.5.3.4 */ + + f = mrb->false_class = mrb_define_class(mrb, "FalseClass", mrb->object_class); + mrb_define_method(mrb, f, "&", false_and, ARGS_REQ(1)); /* 15.2.6.3.1 */ + mrb_define_method(mrb, f, "^", false_xor, ARGS_REQ(1)); /* 15.2.6.3.2 */ + mrb_define_method(mrb, f, "to_s", false_to_s, ARGS_NONE()); /* 15.2.6.3.3 */ + mrb_define_method(mrb, f, "|", false_or, ARGS_REQ(1)); /* 15.2.6.3.4 */ +} + +mrb_value +convert_type(mrb_state *mrb, mrb_value val, const char *tname, const char *method, int raise) +{ + mrb_sym m = 0; + + m = mrb_intern(mrb, method); + if (!mrb_respond_to(mrb, val, m)) { + if (raise) { + mrb_raise(mrb, E_TYPE_ERROR, "can't convert %s into %s", + mrb_nil_p(val) ? "nil" : + (mrb_type(val) == MRB_TT_TRUE) ? "true" : + (mrb_type(val) == MRB_TT_FALSE) ? "false" : + mrb_obj_classname(mrb, val), + tname); + return mrb_nil_value(); + } + else { + return mrb_nil_value(); + } + } + return mrb_funcall(mrb, val, method, 0); +} + +mrb_value +mrb_check_to_integer(mrb_state *mrb, mrb_value val, const char *method) +{ + mrb_value v; + + if (mrb_type(val) == MRB_TT_FIXNUM) return val; + v = convert_type(mrb, val, "Integer", method, FALSE); + if (mrb_nil_p(v)) return (v); + if (!mrb_obj_is_kind_of(mrb, v, mrb_obj_class(mrb, v))) { + return mrb_nil_value(); + } + return v; +} + +mrb_value +mrb_convert_type(mrb_state *mrb, mrb_value val, mrb_int type, const char *tname, const char *method) +{ + mrb_value v; + + if (mrb_type(val) == type) return val; + v = convert_type(mrb, val, tname, method, 1/*Qtrue*/); + if (mrb_type(v) != type) { + mrb_raise(mrb, E_TYPE_ERROR, "%s#%s should return %s", + mrb_obj_classname(mrb, val), method, tname); + } + return v; +} + +mrb_value +mrb_check_convert_type(mrb_state *mrb, mrb_value val, mrb_int type, const char *tname, const char *method) +{ + mrb_value v; + + /* always convert T_DATA */ + if (mrb_type(val) == type && type != MRB_TT_DATA) return val; + v = convert_type(mrb, val, tname, method, 0/*Qfalse*/); + if (mrb_nil_p(v)) return mrb_nil_value(); + if (mrb_type(v) != type) { + mrb_raise(mrb, E_TYPE_ERROR, "%s#%s should return %s", + mrb_obj_classname(mrb, val), method, tname); + } + return v; +} + +static const struct types { + unsigned char type; + const char *name; +} builtin_types[] = { +// {MRB_TT_NIL, "nil"}, + {MRB_TT_FALSE, "false"}, + {MRB_TT_TRUE, "true"}, + {MRB_TT_FIXNUM, "Fixnum"}, + {MRB_TT_SYMBOL, "Symbol"}, /* :symbol */ + {MRB_TT_MODULE, "Module"}, + {MRB_TT_OBJECT, "Object"}, + {MRB_TT_CLASS, "Class"}, + {MRB_TT_ICLASS, "iClass"}, /* internal use: mixed-in module holder */ + {MRB_TT_SCLASS, "SClass"}, + {MRB_TT_PROC, "Proc"}, + {MRB_TT_FLOAT, "Float"}, + {MRB_TT_ARRAY, "Array"}, + {MRB_TT_HASH, "Hash"}, + {MRB_TT_STRING, "String"}, + {MRB_TT_RANGE, "Range"}, + {MRB_TT_REGEX, "Regexp"}, + {MRB_TT_STRUCT, "Struct"}, +// {MRB_TT_BIGNUM, "Bignum"}, + {MRB_TT_FILE, "File"}, + {MRB_TT_DATA, "Data"}, /* internal use: wrapped C pointers */ + {MRB_TT_MATCH, "MatchData"}, /* data of $~ */ +// {MRB_TT_VARMAP, "Varmap"}, /* internal use: dynamic variables */ +// {MRB_TT_NODE, "Node"}, /* internal use: syntax tree node */ +// {MRB_TT_UNDEF, "undef"}, /* internal use: #undef; should not happen */ + {-1, 0} +}; + +void +mrb_check_type(mrb_state *mrb, mrb_value x, enum mrb_vtype t) +{ + const struct types *type = builtin_types; + struct RString *s; + int xt; + + /*if (x == Qundef) { + //mrb_bug("undef leaked to the Ruby space"); + printf ("undef leaked to the Ruby space\n"); + }*/ + + xt = mrb_type(x); + if ((xt != t) || (xt == MRB_TT_DATA)) { + while (type->type < MRB_TT_MAXDEFINE) { + if (type->type == t) { + const char *etype; + + if (mrb_nil_p(x)) { + etype = "nil"; + } + else if (mrb_type(x) == MRB_TT_FIXNUM) { + etype = "Fixnum"; + } + else if (mrb_type(x) == MRB_TT_SYMBOL) { + etype = "Symbol"; + } + else if (mrb_special_const_p(x)) { + s = mrb_str_ptr(mrb_obj_as_string(mrb, x)); + etype = s->buf; + } + else { + etype = mrb_obj_classname(mrb, x); + } + mrb_raise(mrb, E_TYPE_ERROR, "wrong argument type %s (expected %s)", + etype, type->name); + } + type++; + } + /*mrb_bug("unknown type 0x%x", t);*/ + printf ("unknown type 0x%x (0x%x given)", t, mrb_type(x)); + } +} + +/* 15.3.1.3.46 */ +/* + * call-seq: + * obj.to_s => string + * + * Returns a string representing <i>obj</i>. The default + * <code>to_s</code> prints the object's class and an encoding of the + * object id. As a special case, the top-level object that is the + * initial execution context of Ruby programs returns ``main.'' + */ + +mrb_value +mrb_any_to_s(mrb_state *mrb, mrb_value obj) +{ + const char *cname = mrb_obj_classname(mrb, obj); + size_t len; + mrb_value str; + struct RString *s; + + len = strlen(cname)+6+16; + str = mrb_str_new(mrb, 0, len); /* 6:tags 16:addr */ + s = mrb_str_ptr(str); + // snprintf(RSTRING(str)->ptr, len+1, "#<%s:0x%lx>", cname, obj); + sprintf(s->buf, "#<%s:0x%lx>", cname, (unsigned long)(obj.value.p)); + s->len = strlen(s->buf); + /*if (OBJ_TAINTED(obj)) OBJ_TAINT(str);*/ + + return str; +} + +/* + * call-seq: + * obj.is_a?(class) => true or false + * obj.kind_of?(class) => true or false + * + * Returns <code>true</code> if <i>class</i> is the class of + * <i>obj</i>, or if <i>class</i> is one of the superclasses of + * <i>obj</i> or modules included in <i>obj</i>. + * + * module M; end + * class A + * include M + * end + * class B < A; end + * class C < B; end + * b = B.new + * b.instance_of? A #=> false + * b.instance_of? B #=> true + * b.instance_of? C #=> false + * b.instance_of? M #=> false + * b.kind_of? A #=> true + * b.kind_of? B #=> true + * b.kind_of? C #=> false + * b.kind_of? M #=> true + */ + +int +mrb_obj_is_kind_of(mrb_state *mrb, mrb_value obj, struct RClass *c) +{ + struct RClass *cl = mrb_class(mrb, obj); + + switch (c->tt) { + case MRB_TT_MODULE: + case MRB_TT_CLASS: + case MRB_TT_ICLASS: + break; + + default: + mrb_raise(mrb, E_TYPE_ERROR, "class or module required"); + } + + while (cl) { + if (cl == c || cl->mt == c->mt) + return 1/* TRUE */; + cl = cl->super; + } + return 0/* FALSE */; +} + +static mrb_value +mrb_to_integer(mrb_state *mrb, mrb_value val, const char *method) +{ + mrb_value v; + + if (FIXNUM_P(val)) return val; + //if (TYPE(val) == T_BIGNUM) return val; + v = convert_type(mrb, val, "Integer", method, TRUE); + if (!mrb_obj_is_kind_of(mrb, v, mrb->fixnum_class)) { + const char *cname = mrb_obj_classname(mrb, val); + mrb_raise(mrb, E_TYPE_ERROR, "can't convert %s to Integer (%s#%s gives %s)", + cname, cname, method, mrb_obj_classname(mrb, v)); + } + return v; +} + +mrb_value +mrb_to_int(mrb_state *mrb, mrb_value val) +{ + return mrb_to_integer(mrb, val, "to_int"); +} + +static mrb_value +mrb_convert_to_integer(mrb_state *mrb, mrb_value val, int base) +{ + mrb_value tmp; + + if (mrb_nil_p(val)) { + if (base != 0) goto arg_error; + mrb_raise(mrb, E_TYPE_ERROR, "can't convert nil into Integer"); + } + switch (mrb_type(val)) { + case MRB_TT_FLOAT: + if (base != 0) goto arg_error; + if (mrb_float(val) <= (double)FIXNUM_MAX + && mrb_float(val) >= (double)FIXNUM_MIN) { + break; + } + return mrb_dbl2big(mrb, mrb_float(val)); + + case MRB_TT_FIXNUM: + if (base != 0) goto arg_error; + return val; + + case MRB_TT_STRING: +string_conv: + return mrb_str_to_inum(mrb, val, base, TRUE); + + default: + break; + } + if (base != 0) { + tmp = mrb_check_string_type(mrb, val); + if (!mrb_nil_p(tmp)) goto string_conv; + arg_error: + mrb_raise(mrb, E_ARGUMENT_ERROR, "base specified for non string value"); + } + tmp = convert_type(mrb, val, "Integer", "to_int", FALSE); + if (mrb_nil_p(tmp)) { + return mrb_to_integer(mrb, val, "to_i"); + } + return tmp; +} + +mrb_value +mrb_Integer(mrb_state *mrb, mrb_value val) +{ + return mrb_convert_to_integer(mrb, val, 0); +} + +mrb_value +mrb_Float(mrb_state *mrb, mrb_value val) +{ + if (mrb_nil_p(val)) { + mrb_raise(mrb, E_TYPE_ERROR, "can't convert nil into Float"); + } + switch (mrb_type(val)) { + case MRB_TT_FIXNUM: + return mrb_float_value((mrb_float)mrb_fixnum(val)); + + case MRB_TT_FLOAT: + return val; + + case MRB_TT_STRING: + return mrb_float_value(mrb_str_to_dbl(mrb, val, TRUE)); + + default: + return mrb_convert_type(mrb, val, MRB_TT_FLOAT, "Float", "to_f"); + } +} + +mrb_value +mrb_inspect(mrb_state *mrb, mrb_value obj) +{ + return mrb_obj_as_string(mrb, mrb_funcall(mrb, obj, "inspect", 0, 0)); +} + +int +mrb_eql(mrb_state *mrb, mrb_value obj1, mrb_value obj2) +{ + return RTEST(mrb_funcall(mrb, obj1, "eql?", 1, obj2)); +} + |
