summaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authorRory OConnell <[email protected]>2020-07-09 18:52:31 -0700
committerRory OConnell <[email protected]>2020-07-09 18:52:31 -0700
commit338c8538c92a1f3c4117fb920855a610cfaefc25 (patch)
treeaa1498108325d3a5e06323d83d128a1db8d84862
parent81fc5ff48a79c75a7761e1028092b52ad2753cea (diff)
downloadmruby-338c8538c92a1f3c4117fb920855a610cfaefc25.tar.gz
mruby-338c8538c92a1f3c4117fb920855a610cfaefc25.zip
Initial ObjectSpace.memsize_of implementation
-rw-r--r--mrbgems/mruby-objectspace/mrbgem.rake4
-rw-r--r--mrbgems/mruby-objectspace/src/mruby_objectspace.c201
-rw-r--r--mrbgems/mruby-objectspace/test/objectspace.rb76
3 files changed, 281 insertions, 0 deletions
diff --git a/mrbgems/mruby-objectspace/mrbgem.rake b/mrbgems/mruby-objectspace/mrbgem.rake
index fa35136a1..101e24275 100644
--- a/mrbgems/mruby-objectspace/mrbgem.rake
+++ b/mrbgems/mruby-objectspace/mrbgem.rake
@@ -2,4 +2,8 @@ MRuby::Gem::Specification.new('mruby-objectspace') do |spec|
spec.license = 'MIT'
spec.author = 'mruby developers'
spec.summary = 'ObjectSpace class'
+
+ spec.add_test_dependency('mruby-metaprog')
+ spec.add_test_dependency('mruby-method')
+ spec.add_test_dependency('mruby-fiber')
end
diff --git a/mrbgems/mruby-objectspace/src/mruby_objectspace.c b/mrbgems/mruby-objectspace/src/mruby_objectspace.c
index b89fb0580..fe7e929c8 100644
--- a/mrbgems/mruby-objectspace/src/mruby_objectspace.c
+++ b/mrbgems/mruby-objectspace/src/mruby_objectspace.c
@@ -2,6 +2,14 @@
#include <mruby/gc.h>
#include <mruby/hash.h>
#include <mruby/class.h>
+#include <mruby/object.h>
+#include <mruby/numeric.h>
+#include <mruby/string.h>
+#include <mruby/array.h>
+#include <mruby/variable.h>
+#include <mruby/proc.h>
+#include <mruby/value.h>
+#include <mruby/range.h>
struct os_count_struct {
mrb_int total;
@@ -168,12 +176,205 @@ os_each_object(mrb_state *mrb, mrb_value self)
return mrb_fixnum_value(d.count);
}
+static void os_memsize_of_object(mrb_state*,mrb_value,mrb_bool,mrb_int*);
+
+static int
+os_memsize_ivar_cb(mrb_state *mrb, mrb_sym _name, mrb_value obj, void *data)
+{
+ mrb_int *cb_data = (mrb_int *)data;
+ mrb_int recurse = *(&cb_data[0]);
+ mrb_int* total = &cb_data[1];
+
+ os_memsize_of_object(mrb, obj, (mrb_bool)(recurse), total);
+ return 0;
+}
+
+static void
+os_memsize_of_ivars(mrb_state* mrb, mrb_value obj, mrb_bool recurse, mrb_int *t)
+{
+ /* need iv segment table size */
+ if(recurse) {
+ mrb_int r = (mrb_int)recurse;
+ mrb_int *cb_data[2] = { &r, t };
+ mrb_iv_foreach(mrb, obj, os_memsize_ivar_cb, t);
+ }
+}
+
+static void
+os_memsize_of_irep(mrb_state* state, struct mrb_irep *irep, mrb_int* t)
+{
+ mrb_int i;
+ (*t) += (irep->slen * sizeof(mrb_sym)) +
+ (irep->plen * sizeof(mrb_code)) +
+ (irep->ilen * sizeof(mrb_code));
+
+ for(i = 0; i < irep->rlen; i++) {
+ os_memsize_of_irep(state, irep->reps[i], t);
+ }
+}
+
+static void
+os_memsize_of_method(mrb_state* mrb, mrb_value method_obj, mrb_int* t)
+{
+ mrb_value proc_value = mrb_obj_iv_get(mrb, mrb_obj_ptr(method_obj),
+ mrb_intern_lit(mrb, "_proc"));
+ struct RProc *proc = mrb_proc_ptr(proc_value);
+
+ (*t) += sizeof(struct RProc);
+ if(!MRB_PROC_CFUNC_P(proc)) os_memsize_of_irep(mrb, proc->body.irep, t);
+}
+
+static void
+os_memsize_of_methods(mrb_state* mrb, mrb_value obj, mrb_int* t)
+{
+ mrb_value method_list;
+ mrb_int i;
+ if(!mrb_respond_to(mrb, obj, mrb_intern_lit(mrb, "instance_methods"))) return;
+ method_list = mrb_funcall(mrb, obj, "instance_methods", 1, mrb_false_value());
+ for(i = 0; i < RARRAY_LEN(method_list); i++) {
+ mrb_value method = mrb_funcall(mrb, obj, "instance_method", 1,
+ mrb_ary_ref(mrb, method_list, i));
+ os_memsize_of_method(mrb, method, t);
+ }
+}
+
+static void
+os_memsize_of_object(mrb_state* mrb, mrb_value obj, mrb_bool recurse, mrb_int* t)
+{
+ switch(obj.tt) {
+ case MRB_TT_STRING:
+ (*t) += RSTRING_LEN(obj);
+ break;
+ case MRB_TT_CLASS:
+ case MRB_TT_MODULE:
+ case MRB_TT_EXCEPTION:
+ case MRB_TT_SCLASS:
+ case MRB_TT_ICLASS:
+ case MRB_TT_OBJECT: {
+ os_memsize_of_ivars(mrb, obj, recurse, t);
+ if(mrb_obj_is_kind_of(mrb, obj, mrb_class_get(mrb, "UnboundMethod"))) {
+ os_memsize_of_method(mrb, obj, t);
+ }
+ else {
+ os_memsize_of_methods(mrb, obj, t);
+ }
+ break;
+ }
+ case MRB_TT_HASH: {
+ struct htable* htable = RHASH_TBL(obj);
+ /* Need htable & segment struct defs */
+ break;
+ }
+ case MRB_TT_ARRAY: {
+ mrb_int len, i;
+ len = RARRAY_LEN(obj);
+ /* Arrays that do not fit within an RArray perform a heap allocation
+ * storing an array of pointers to the original objects*/
+ if(len > MRB_ARY_EMBED_LEN_MAX) (*t) += sizeof(mrb_value *) * len;
+
+ if(recurse) {
+ for(i = 0; i < len; i++) {
+ os_memsize_of_object(mrb, ARY_PTR(mrb_ary_ptr(obj))[i], recurse, t);
+ }
+ }
+ break;
+ }
+ case MRB_TT_PROC: {
+ struct RProc* proc = mrb_proc_ptr(obj);
+ (*t) += MRB_ENV_LEN(proc->e.env) * sizeof(mrb_value);
+ if(!MRB_PROC_CFUNC_P(proc)) os_memsize_of_irep(mrb, proc->body.irep, t);
+ break;
+ }
+ case MRB_TT_DATA:
+ if(mrb_respond_to(mrb, obj, mrb_intern_lit(mrb, "memsize"))) {
+ (*t) += mrb_fixnum(mrb_funcall(mrb, obj, "memsize", 0));
+ }
+ break;
+ #ifndef MRB_WITHOUT_FLOAT
+ case MRB_TT_FLOAT:
+ #ifdef MRB_WORD_BOXING
+ (*t) += sizeof(struct RFloat);
+ #endif
+ break;
+ #endif
+ case MRB_TT_RANGE:
+ #ifndef MRB_RANGE_EMBED
+ (*t) += sizeof(struct mrb_range_edges);
+ #endif
+ break;
+ case MRB_TT_FIBER:
+ struct RFiber* fiber = (struct RFiber*)mrb_ptr(obj);
+ (*t) += sizeof(struct mrb_context);
+ break;
+ /* zero heap size types.
+ * immediate VM stack values, contained within mrb_state, mrb_heap_page,
+ * or on C stack */
+ case MRB_TT_TRUE:
+ case MRB_TT_FALSE:
+ case MRB_TT_FIXNUM:
+ case MRB_TT_BREAK:
+ case MRB_TT_CPTR:
+ case MRB_TT_SYMBOL:
+ case MRB_TT_FREE:
+ case MRB_TT_UNDEF:
+ case MRB_TT_ENV:
+ case MRB_TT_ISTRUCT:
+ /* never used, silences compiler warning
+ * not having a default: clause lets the compiler tell us when there is a new
+ * TT not accounted for */
+ case MRB_TT_MAXDEFINE:
+ break;
+ }
+}
+
+/*
+ * call-seq:
+ * ObjectSpace.memsize_of(obj, recurse: false) -> Numeric
+ *
+ * Returns the amount of heap memory allocated for object in size_t units.
+ * Not all objects cause additional heap allocations beyond their object pointer
+ * in the heap page and may return 0.
+ *
+ * The return value depends on the definition of size_t on that platform,
+ * therefore the value is not comparable across platform types.
+ *
+ * Immediate values such as integers, booleans, symbols and unboxed float numbers
+ * return 0. Additionally special objects which are small enough to fit inside an
+ * object * pointer, termed embedded objects, also return 0. Strings and arrays
+ * below a compile-time defined size may be embedded.
+ *
+ * Setting recurse: true descends into instance variables, array members,
+ * and hash values recursively, calculating the child objects and adding to
+ * the final sum.
+ *
+ */
+
+static mrb_value
+os_memsize_of(mrb_state *mrb, mrb_value self)
+{
+ mrb_int total;
+ mrb_value obj;
+ mrb_bool recurse;
+ const char *kw_names[1] = { "recurse" };
+ mrb_value kw_values[1];
+ const mrb_kwargs kwargs = { 1, kw_values, kw_names, 0, NULL };
+
+ mrb_get_args(mrb, "o:", &obj, &kwargs);
+ recurse = mrb_obj_eq(mrb, kw_values[0], mrb_true_value()) ? TRUE : FALSE;
+
+ total = 0;
+ os_memsize_of_object(mrb, obj, recurse, &total);
+
+ return mrb_fixnum_value(total);
+}
+
void
mrb_mruby_objectspace_gem_init(mrb_state *mrb)
{
struct RClass *os = mrb_define_module(mrb, "ObjectSpace");
mrb_define_class_method(mrb, os, "count_objects", os_count_objects, MRB_ARGS_OPT(1));
mrb_define_class_method(mrb, os, "each_object", os_each_object, MRB_ARGS_OPT(1));
+ mrb_define_class_method(mrb, os, "memsize_of", os_memsize_of, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1));
}
void
diff --git a/mrbgems/mruby-objectspace/test/objectspace.rb b/mrbgems/mruby-objectspace/test/objectspace.rb
index 9c44c2157..d26fd5a9e 100644
--- a/mrbgems/mruby-objectspace/test/objectspace.rb
+++ b/mrbgems/mruby-objectspace/test/objectspace.rb
@@ -1,3 +1,4 @@
+# coding: utf-8
assert('ObjectSpace.count_objects') do
h = {}
f = Fiber.new {} if Object.const_defined?(:Fiber)
@@ -58,3 +59,78 @@ end
assert 'Check class pointer of ObjectSpace.each_object.' do
assert_nothing_raised { ObjectSpace.each_object { |obj| !obj } }
end
+
+assert 'ObjectSpace.memsize_of' do
+ # immediate literals
+ int_size = ObjectSpace.memsize_of 1
+ assert_equal int_size, 0, 'int zero'
+
+ sym_size = ObjectSpace.memsize_of :foo
+ assert_equal sym_size, 0, 'sym zero'
+
+ assert_equal ObjectSpace.memsize_of(true), int_size
+ assert_equal ObjectSpace.memsize_of(false), int_size
+
+ float_size = if Object.const_defined? :Float
+ ObjectSpace.memsize_of 1.0
+ else
+ nil
+ end
+
+ # need some way of asking if floats are boxed
+ assert_equal float_size, 0 if float_size
+
+ assert_not_equal ObjectSpace.memsize_of('a'), 0, 'memsize of str'
+
+ if __ENCODING__ == "UTF-8"
+ assert_not_equal ObjectSpace.memsize_of("こんにちは世界"), 0, 'memsize of utf8 str'
+ end
+
+ assert_not_equal ObjectSpace.memsize_of(0..1), 0, 'range not zero'
+
+ # class defs
+ class_obj_size = ObjectSpace.memsize_of Class
+ assert_not_equal class_obj_size, 0, 'Class obj not zero'
+
+ empty_class_def_size = ObjectSpace.memsize_of Class.new
+
+ # need access to struct iv_tbl
+ # assert_not_equal empty_class_def_size, 0, 'Class def not zero'
+
+ class_without_methods = Class.new do
+ @a = 1
+ @b = 2
+ end
+ class_total_size = empty_class_def_size + (int_size * 2)
+ assert_equal ObjectSpace.memsize_of(class_without_methods), class_total_size, 'class without methods size'
+
+ module_without_methods = Module.new do
+ @a = 1
+ @b = 2
+ end
+ module_total_size = empty_class_def_size + (int_size * 2)
+ assert_equal ObjectSpace.memsize_of(module_without_methods), module_total_size, 'module without methods size'
+
+ proc_size = ObjectSpace.memsize_of Proc.new { x = 1; x }
+ assert_not_equal proc_size, 0
+
+ class_with_methods = Class.new do
+ def foo
+ a = 0
+ a + 1
+ end
+ end
+
+ m_size = ObjectSpace.memsize_of class_with_methods.instance_method(:foo)
+ assert_not_equal m_size, 0, 'method size not zero'
+
+ # collections
+ assert_equal ObjectSpace.memsize_of([]), 0, 'empty array size zero'
+ assert_not_equal ObjectSpace.memsize_of(Array.new(16)), 0, 'array size non zero'
+
+ # fiber
+ assert_not_equal ObjectSpace.memsize_of(Fiber.new {}), 0, 'fiber non zero'
+
+ skip 'No hash table support yet'
+ assert_equal ObjectSpace.memsize_of({}), 0, 'empty hash size zero'
+end