mbed I/F binding for mruby

Dependents:   mruby_mbed_web mirb_mbed

mbed-mruby

How to use

Class

Committer:
mzta
Date:
Wed Mar 25 17:36:16 2015 +0000
Revision:
0:158c61bb030f
mirb_mbed initial commit;

Who changed what in which revision?

UserRevisionLine numberNew contents of line
mzta 0:158c61bb030f 1 #include "mruby.h"
mzta 0:158c61bb030f 2 #include "mruby/gc.h"
mzta 0:158c61bb030f 3 #include "mruby/hash.h"
mzta 0:158c61bb030f 4 #include "mruby/class.h"
mzta 0:158c61bb030f 5
mzta 0:158c61bb030f 6 struct os_count_struct {
mzta 0:158c61bb030f 7 mrb_int total;
mzta 0:158c61bb030f 8 mrb_int freed;
mzta 0:158c61bb030f 9 mrb_int counts[MRB_TT_MAXDEFINE+1];
mzta 0:158c61bb030f 10 };
mzta 0:158c61bb030f 11
mzta 0:158c61bb030f 12 static void
mzta 0:158c61bb030f 13 os_count_object_type(mrb_state *mrb, struct RBasic *obj, void *data)
mzta 0:158c61bb030f 14 {
mzta 0:158c61bb030f 15 struct os_count_struct *obj_count;
mzta 0:158c61bb030f 16 obj_count = (struct os_count_struct*)data;
mzta 0:158c61bb030f 17
mzta 0:158c61bb030f 18 obj_count->total++;
mzta 0:158c61bb030f 19
mzta 0:158c61bb030f 20 if (is_dead(mrb, obj)) {
mzta 0:158c61bb030f 21 obj_count->freed++;
mzta 0:158c61bb030f 22 }
mzta 0:158c61bb030f 23 else {
mzta 0:158c61bb030f 24 obj_count->counts[obj->tt]++;
mzta 0:158c61bb030f 25 }
mzta 0:158c61bb030f 26 }
mzta 0:158c61bb030f 27
mzta 0:158c61bb030f 28 /*
mzta 0:158c61bb030f 29 * call-seq:
mzta 0:158c61bb030f 30 * ObjectSpace.count_objects([result_hash]) -> hash
mzta 0:158c61bb030f 31 *
mzta 0:158c61bb030f 32 * Counts objects for each type.
mzta 0:158c61bb030f 33 *
mzta 0:158c61bb030f 34 * It returns a hash, such as:
mzta 0:158c61bb030f 35 * {
mzta 0:158c61bb030f 36 * :TOTAL=>10000,
mzta 0:158c61bb030f 37 * :FREE=>3011,
mzta 0:158c61bb030f 38 * :T_OBJECT=>6,
mzta 0:158c61bb030f 39 * :T_CLASS=>404,
mzta 0:158c61bb030f 40 * # ...
mzta 0:158c61bb030f 41 * }
mzta 0:158c61bb030f 42 *
mzta 0:158c61bb030f 43 * If the optional argument +result_hash+ is given,
mzta 0:158c61bb030f 44 * it is overwritten and returned. This is intended to avoid probe effect.
mzta 0:158c61bb030f 45 *
mzta 0:158c61bb030f 46 */
mzta 0:158c61bb030f 47
mzta 0:158c61bb030f 48 static mrb_value
mzta 0:158c61bb030f 49 os_count_objects(mrb_state *mrb, mrb_value self)
mzta 0:158c61bb030f 50 {
mzta 0:158c61bb030f 51 struct os_count_struct obj_count = { 0 };
mzta 0:158c61bb030f 52 enum mrb_vtype i;
mzta 0:158c61bb030f 53 mrb_value hash;
mzta 0:158c61bb030f 54
mzta 0:158c61bb030f 55 if (mrb_get_args(mrb, "|H", &hash) == 0) {
mzta 0:158c61bb030f 56 hash = mrb_hash_new(mrb);
mzta 0:158c61bb030f 57 }
mzta 0:158c61bb030f 58
mzta 0:158c61bb030f 59 if (!mrb_test(mrb_hash_empty_p(mrb, hash))) {
mzta 0:158c61bb030f 60 mrb_hash_clear(mrb, hash);
mzta 0:158c61bb030f 61 }
mzta 0:158c61bb030f 62
mzta 0:158c61bb030f 63 mrb_objspace_each_objects(mrb, os_count_object_type, &obj_count);
mzta 0:158c61bb030f 64
mzta 0:158c61bb030f 65 mrb_hash_set(mrb, hash, mrb_symbol_value(mrb_intern_lit(mrb, "TOTAL")), mrb_fixnum_value(obj_count.total));
mzta 0:158c61bb030f 66 mrb_hash_set(mrb, hash, mrb_symbol_value(mrb_intern_lit(mrb, "FREE")), mrb_fixnum_value(obj_count.freed));
mzta 0:158c61bb030f 67
mzta 0:158c61bb030f 68 for (i = MRB_TT_FALSE; i < MRB_TT_MAXDEFINE; i++) {
mzta 0:158c61bb030f 69 mrb_value type;
mzta 0:158c61bb030f 70 switch (i) {
mzta 0:158c61bb030f 71 #define COUNT_TYPE(t) case (MRB_T ## t): type = mrb_symbol_value(mrb_intern_lit(mrb, #t)); break;
mzta 0:158c61bb030f 72 COUNT_TYPE(T_FALSE);
mzta 0:158c61bb030f 73 COUNT_TYPE(T_FREE);
mzta 0:158c61bb030f 74 COUNT_TYPE(T_TRUE);
mzta 0:158c61bb030f 75 COUNT_TYPE(T_FIXNUM);
mzta 0:158c61bb030f 76 COUNT_TYPE(T_SYMBOL);
mzta 0:158c61bb030f 77 COUNT_TYPE(T_UNDEF);
mzta 0:158c61bb030f 78 COUNT_TYPE(T_FLOAT);
mzta 0:158c61bb030f 79 COUNT_TYPE(T_CPTR);
mzta 0:158c61bb030f 80 COUNT_TYPE(T_OBJECT);
mzta 0:158c61bb030f 81 COUNT_TYPE(T_CLASS);
mzta 0:158c61bb030f 82 COUNT_TYPE(T_MODULE);
mzta 0:158c61bb030f 83 COUNT_TYPE(T_ICLASS);
mzta 0:158c61bb030f 84 COUNT_TYPE(T_SCLASS);
mzta 0:158c61bb030f 85 COUNT_TYPE(T_PROC);
mzta 0:158c61bb030f 86 COUNT_TYPE(T_ARRAY);
mzta 0:158c61bb030f 87 COUNT_TYPE(T_HASH);
mzta 0:158c61bb030f 88 COUNT_TYPE(T_STRING);
mzta 0:158c61bb030f 89 COUNT_TYPE(T_RANGE);
mzta 0:158c61bb030f 90 COUNT_TYPE(T_EXCEPTION);
mzta 0:158c61bb030f 91 COUNT_TYPE(T_FILE);
mzta 0:158c61bb030f 92 COUNT_TYPE(T_ENV);
mzta 0:158c61bb030f 93 COUNT_TYPE(T_DATA);
mzta 0:158c61bb030f 94 COUNT_TYPE(T_FIBER);
mzta 0:158c61bb030f 95 #undef COUNT_TYPE
mzta 0:158c61bb030f 96 default:
mzta 0:158c61bb030f 97 type = mrb_fixnum_value(i); break;
mzta 0:158c61bb030f 98 }
mzta 0:158c61bb030f 99 if (obj_count.counts[i])
mzta 0:158c61bb030f 100 mrb_hash_set(mrb, hash, type, mrb_fixnum_value(obj_count.counts[i]));
mzta 0:158c61bb030f 101 }
mzta 0:158c61bb030f 102
mzta 0:158c61bb030f 103 return hash;
mzta 0:158c61bb030f 104 }
mzta 0:158c61bb030f 105
mzta 0:158c61bb030f 106 struct os_each_object_data {
mzta 0:158c61bb030f 107 mrb_value block;
mzta 0:158c61bb030f 108 struct RClass *target_module;
mzta 0:158c61bb030f 109 mrb_int count;
mzta 0:158c61bb030f 110 };
mzta 0:158c61bb030f 111
mzta 0:158c61bb030f 112 static void
mzta 0:158c61bb030f 113 os_each_object_cb(mrb_state *mrb, struct RBasic *obj, void *ud)
mzta 0:158c61bb030f 114 {
mzta 0:158c61bb030f 115 struct os_each_object_data *d = (struct os_each_object_data*)ud;
mzta 0:158c61bb030f 116
mzta 0:158c61bb030f 117 /* filter dead objects */
mzta 0:158c61bb030f 118 if (is_dead(mrb, obj)) {
mzta 0:158c61bb030f 119 return;
mzta 0:158c61bb030f 120 }
mzta 0:158c61bb030f 121
mzta 0:158c61bb030f 122 /* filter internal objects */
mzta 0:158c61bb030f 123 switch (obj->tt) {
mzta 0:158c61bb030f 124 case MRB_TT_ENV:
mzta 0:158c61bb030f 125 case MRB_TT_ICLASS:
mzta 0:158c61bb030f 126 return;
mzta 0:158c61bb030f 127 default:
mzta 0:158c61bb030f 128 break;
mzta 0:158c61bb030f 129 }
mzta 0:158c61bb030f 130
mzta 0:158c61bb030f 131 /* filter half baked (or internal) objects */
mzta 0:158c61bb030f 132 if (!obj->c) return;
mzta 0:158c61bb030f 133
mzta 0:158c61bb030f 134 /* filter class kind if target module defined */
mzta 0:158c61bb030f 135 if (d->target_module && !mrb_obj_is_kind_of(mrb, mrb_obj_value(obj), d->target_module)) {
mzta 0:158c61bb030f 136 return;
mzta 0:158c61bb030f 137 }
mzta 0:158c61bb030f 138
mzta 0:158c61bb030f 139 mrb_yield(mrb, d->block, mrb_obj_value(obj));
mzta 0:158c61bb030f 140 ++d->count;
mzta 0:158c61bb030f 141 }
mzta 0:158c61bb030f 142
mzta 0:158c61bb030f 143 /*
mzta 0:158c61bb030f 144 * call-seq:
mzta 0:158c61bb030f 145 * ObjectSpace.each_object([module]) {|obj| ... } -> fixnum
mzta 0:158c61bb030f 146 *
mzta 0:158c61bb030f 147 * Calls the block once for each object in this Ruby process.
mzta 0:158c61bb030f 148 * Returns the number of objects found.
mzta 0:158c61bb030f 149 * If the optional argument +module+ is given,
mzta 0:158c61bb030f 150 * calls the block for only those classes or modules
mzta 0:158c61bb030f 151 * that match (or are a subclass of) +module+.
mzta 0:158c61bb030f 152 *
mzta 0:158c61bb030f 153 * If no block is given, ArgumentError is raised.
mzta 0:158c61bb030f 154 *
mzta 0:158c61bb030f 155 */
mzta 0:158c61bb030f 156
mzta 0:158c61bb030f 157 static mrb_value
mzta 0:158c61bb030f 158 os_each_object(mrb_state *mrb, mrb_value self)
mzta 0:158c61bb030f 159 {
mzta 0:158c61bb030f 160 mrb_value cls = mrb_nil_value();
mzta 0:158c61bb030f 161 struct os_each_object_data d;
mzta 0:158c61bb030f 162 mrb_get_args(mrb, "&|C", &d.block, &cls);
mzta 0:158c61bb030f 163
mzta 0:158c61bb030f 164 if (mrb_nil_p(d.block)) {
mzta 0:158c61bb030f 165 mrb_raise(mrb, E_ARGUMENT_ERROR, "Expected block in ObjectSpace.each_object.");
mzta 0:158c61bb030f 166 }
mzta 0:158c61bb030f 167
mzta 0:158c61bb030f 168 d.target_module = mrb_nil_p(cls) ? NULL : mrb_class_ptr(cls);
mzta 0:158c61bb030f 169 d.count = 0;
mzta 0:158c61bb030f 170 mrb_objspace_each_objects(mrb, os_each_object_cb, &d);
mzta 0:158c61bb030f 171 return mrb_fixnum_value(d.count);
mzta 0:158c61bb030f 172 }
mzta 0:158c61bb030f 173
mzta 0:158c61bb030f 174 void
mzta 0:158c61bb030f 175 mrb_mruby_objectspace_gem_init(mrb_state *mrb)
mzta 0:158c61bb030f 176 {
mzta 0:158c61bb030f 177 struct RClass *os = mrb_define_module(mrb, "ObjectSpace");
mzta 0:158c61bb030f 178 mrb_define_class_method(mrb, os, "count_objects", os_count_objects, MRB_ARGS_OPT(1));
mzta 0:158c61bb030f 179 mrb_define_class_method(mrb, os, "each_object", os_each_object, MRB_ARGS_OPT(1));
mzta 0:158c61bb030f 180 }
mzta 0:158c61bb030f 181
mzta 0:158c61bb030f 182 void
mzta 0:158c61bb030f 183 mrb_mruby_objectspace_gem_final(mrb_state *mrb)
mzta 0:158c61bb030f 184 {
mzta 0:158c61bb030f 185 }
mzta 0:158c61bb030f 186