mirror of
				https://github.com/ruby/ruby.git
				synced 2022-11-09 12:17:21 -05:00 
			
		
		
		
	* ext/objspace/objspace.c: support ObjectSpace.trace_object_allocations.
Read the following test to know HOWTO. This feature is a sample of RUBY_INTERNAL_EVENT. * test/objspace/test_objspace.rb: add a test. * ext/objspace/object_tracing.c: ditto. * gc.c (rb_gc_count): add. THis function returns GC count. * internal.h: add decl. of rb_gc_count(). Same as `GC.count'. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@40957 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
		
							parent
							
								
									14f6c8ca8a
								
							
						
					
					
						commit
						050dd10d6f
					
				
					 6 changed files with 242 additions and 1 deletions
				
			
		
							
								
								
									
										14
									
								
								ChangeLog
									
										
									
									
									
								
							
							
						
						
									
										14
									
								
								ChangeLog
									
										
									
									
									
								
							|  | @ -1,3 +1,17 @@ | |||
| Mon May 27 18:57:28 2013  Koichi Sasada  <ko1@atdot.net> | ||||
| 
 | ||||
| 	* ext/objspace/objspace.c: support ObjectSpace.trace_object_allocations. | ||||
| 	  Read the following test to know HOWTO. | ||||
| 	  This feature is a sample of RUBY_INTERNAL_EVENT. | ||||
| 
 | ||||
| 	* test/objspace/test_objspace.rb: add a test. | ||||
| 
 | ||||
| 	* ext/objspace/object_tracing.c: ditto. | ||||
| 
 | ||||
| 	* gc.c (rb_gc_count): add. THis function returns GC count. | ||||
| 
 | ||||
| 	* internal.h: add decl. of rb_gc_count(). Same as `GC.count'. | ||||
| 
 | ||||
| Mon May 27 17:33:28 2013  Nobuyoshi Nakada  <nobu@ruby-lang.org> | ||||
| 
 | ||||
| 	* tool/rbinstall.rb (install_recursive): add maxdepth option. | ||||
|  |  | |||
							
								
								
									
										196
									
								
								ext/objspace/object_tracing.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										196
									
								
								ext/objspace/object_tracing.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,196 @@ | |||
| /**********************************************************************
 | ||||
| 
 | ||||
|   object_traceing.c - Object Tracing mechanism/ObjectSpace extender for MRI. | ||||
| 
 | ||||
|   $Author: sorah $ | ||||
|   created at: Mon May 27 16:27:44 2013 | ||||
| 
 | ||||
|   NOTE: This extension library is not expected to exist except C Ruby. | ||||
|   NOTE: This feature is an example usage of internal event tracing APIs. | ||||
| 
 | ||||
|   All the files in this distribution are covered under the Ruby's | ||||
|   license (see the file COPYING). | ||||
| 
 | ||||
| **********************************************************************/ | ||||
| 
 | ||||
| #include "ruby/ruby.h" | ||||
| #include "ruby/debug.h" | ||||
| 
 | ||||
| size_t rb_gc_count(void); /* from gc.c */ | ||||
| 
 | ||||
| struct traceobj_arg { | ||||
|     VALUE newobj_trace; | ||||
|     VALUE freeobj_trace; | ||||
|     st_table *object_table; | ||||
|     st_table *path_table; | ||||
|     struct traceobj_arg *prev_traceobj_arg; | ||||
| }; | ||||
| 
 | ||||
| struct traceobj_arg *traceobj_arg; /* TODO: do not use GLOBAL VARIABLE!!! */ | ||||
| 
 | ||||
| struct allocation_info { | ||||
|     char *path; | ||||
|     unsigned long line; | ||||
|     size_t generation; | ||||
| }; | ||||
| 
 | ||||
| static void | ||||
| newobj_i(VALUE tpval, void *data) | ||||
| { | ||||
|     struct traceobj_arg *arg = (struct traceobj_arg *)data; | ||||
|     rb_trace_arg_t *tparg = rb_tracearg_from_tracepoint(tpval); | ||||
|     VALUE obj = rb_tracearg_object(tparg); | ||||
|     VALUE path = rb_tracearg_path(tparg); | ||||
|     VALUE line = rb_tracearg_lineno(tparg); | ||||
|     int path_len = RSTRING_LEN(path); | ||||
|     struct allocation_info *info = (struct allocation_info *)ruby_xmalloc(sizeof(struct allocation_info)); | ||||
|     char *path_cstr = ruby_xmalloc(path_len + 1); | ||||
|     char *path_stored_cstr; | ||||
| 
 | ||||
|     strncpy(path_cstr, RSTRING_PTR(path), path_len); | ||||
|     path_cstr[path_len] = 0; | ||||
| 
 | ||||
|     if (st_get_key(arg->path_table, (st_data_t)path_cstr, (st_data_t *)&path_stored_cstr)) { | ||||
| 	st_data_t n; | ||||
| 	st_lookup(arg->path_table, (st_data_t)path_stored_cstr, &n); | ||||
| 	st_insert(arg->path_table, (st_data_t)path_stored_cstr, n+1); | ||||
| 	ruby_xfree(path_cstr); | ||||
| 	path_cstr = path_stored_cstr; | ||||
|     } | ||||
|     else { | ||||
| 	st_add_direct(arg->path_table, (st_data_t)path_cstr, 1); | ||||
|     } | ||||
| 
 | ||||
|     info->path = path_cstr; | ||||
|     info->line = NUM2INT(line); | ||||
|     info->generation = rb_gc_count(); | ||||
|     st_insert(arg->path_table, (st_data_t)path_cstr, 0); | ||||
|     st_insert(arg->object_table, (st_data_t)obj, (st_data_t)info); | ||||
| } | ||||
| 
 | ||||
| static void | ||||
| freeobj_i(VALUE tpval, void *data) | ||||
| { | ||||
|     struct traceobj_arg *arg = (struct traceobj_arg *)data; | ||||
|     rb_trace_arg_t *tparg = rb_tracearg_from_tracepoint(tpval); | ||||
|     VALUE obj = rb_tracearg_object(tparg); | ||||
|     struct allocation_info *info; | ||||
|     st_data_t n; | ||||
|     if (st_delete(arg->object_table, (st_data_t *)&obj, (st_data_t *)&info)) { | ||||
| 	st_lookup(arg->path_table, (st_data_t)info->path, &n); | ||||
| 	if (n == 1) { | ||||
| 	    st_delete(arg->path_table, (st_data_t *)&info->path, 0); | ||||
| 	    ruby_xfree(info->path); | ||||
| 	} | ||||
| 	else { | ||||
| 	    st_insert(arg->path_table, (st_data_t)info->path, n-1); | ||||
| 	} | ||||
| 
 | ||||
| 	ruby_xfree(info); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| static int | ||||
| free_keys_i(st_data_t key, st_data_t value, void *data) | ||||
| { | ||||
|     ruby_xfree((void *)key); | ||||
|     return ST_CONTINUE; | ||||
| } | ||||
| 
 | ||||
| static int | ||||
| free_values_i(st_data_t key, st_data_t value, void *data) | ||||
| { | ||||
|     ruby_xfree((void *)value); | ||||
|     return ST_CONTINUE; | ||||
| } | ||||
| 
 | ||||
| static VALUE | ||||
| stop_trace_object_allocations(void *data) | ||||
| { | ||||
|     struct traceobj_arg *arg = (struct traceobj_arg *)data; | ||||
|     rb_tracepoint_disable(arg->newobj_trace); | ||||
|     rb_tracepoint_disable(arg->freeobj_trace); | ||||
|     st_foreach(arg->object_table, free_values_i, 0); | ||||
|     st_foreach(arg->path_table, free_keys_i, 0); | ||||
|     st_free_table(arg->object_table); | ||||
|     st_free_table(arg->path_table); | ||||
|     traceobj_arg = arg->prev_traceobj_arg; | ||||
| 
 | ||||
|     return Qnil; | ||||
| } | ||||
| 
 | ||||
| static VALUE | ||||
| trace_object_allocations(VALUE objspace) | ||||
| { | ||||
|     struct traceobj_arg arg; | ||||
| 
 | ||||
|     arg.newobj_trace = rb_tracepoint_new(0, RUBY_INTERNAL_EVENT_NEWOBJ, newobj_i, &arg); | ||||
|     arg.freeobj_trace = rb_tracepoint_new(0, RUBY_INTERNAL_EVENT_FREE, freeobj_i, &arg); | ||||
|     arg.object_table = st_init_numtable(); | ||||
|     arg.path_table = st_init_strtable(); | ||||
| 
 | ||||
|     arg.prev_traceobj_arg = traceobj_arg; | ||||
|     traceobj_arg = &arg; | ||||
| 
 | ||||
|     rb_tracepoint_enable(arg.newobj_trace); | ||||
|     rb_tracepoint_enable(arg.freeobj_trace); | ||||
| 
 | ||||
|     return rb_ensure(rb_yield, Qnil, stop_trace_object_allocations, (VALUE)&arg); | ||||
| } | ||||
| 
 | ||||
| struct allocation_info * | ||||
| allocation_info(VALUE obj) | ||||
| { | ||||
|     if (traceobj_arg) { | ||||
| 	struct allocation_info *info; | ||||
| 	if (st_lookup(traceobj_arg->object_table, obj, (st_data_t *)&info)) { | ||||
| 	    return info; | ||||
| 	} | ||||
|     } | ||||
|     return NULL; | ||||
| } | ||||
| 
 | ||||
| static VALUE | ||||
| allocation_sourcefile(VALUE objspace, VALUE obj) | ||||
| { | ||||
|     struct allocation_info *info = allocation_info(obj); | ||||
|     if (info) { | ||||
| 	return rb_str_new2(info->path); | ||||
|     } | ||||
|     else { | ||||
| 	return Qnil; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| static VALUE | ||||
| allocation_sourceline(VALUE objspace, VALUE obj) | ||||
| { | ||||
|     struct allocation_info *info = allocation_info(obj); | ||||
|     if (info) { | ||||
| 	return INT2FIX(info->line); | ||||
|     } | ||||
|     else { | ||||
| 	return Qnil; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| static VALUE | ||||
| allocation_generation(VALUE objspace, VALUE obj) | ||||
| { | ||||
|     struct allocation_info *info = allocation_info(obj); | ||||
|     if (info) { | ||||
| 	return SIZET2NUM(info->generation); | ||||
|     } | ||||
|     else { | ||||
| 	return Qnil; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void | ||||
| Init_object_tracing(VALUE rb_mObjSpace) | ||||
| { | ||||
|     rb_define_module_function(rb_mObjSpace, "trace_object_allocations", trace_object_allocations, 0); | ||||
|     rb_define_module_function(rb_mObjSpace, "allocation_sourcefile", allocation_sourcefile, 1); | ||||
|     rb_define_module_function(rb_mObjSpace, "allocation_sourceline", allocation_sourceline, 1); | ||||
|     rb_define_module_function(rb_mObjSpace, "allocation_generation", allocation_generation, 1); | ||||
| } | ||||
|  | @ -779,6 +779,8 @@ reachable_objects_from(VALUE self, VALUE obj) | |||
|     } | ||||
| } | ||||
| 
 | ||||
| void Init_object_tracing(VALUE rb_mObjSpace); | ||||
| 
 | ||||
| /* objspace library extends ObjectSpace module and add several
 | ||||
|  * methods to get internal statistic information about | ||||
|  * object/memory management. | ||||
|  | @ -807,4 +809,6 @@ Init_objspace(void) | |||
|     rb_define_method(rb_mInternalObjectWrapper, "type", iow_type, 0); | ||||
|     rb_define_method(rb_mInternalObjectWrapper, "inspect", iow_inspect, 0); | ||||
|     rb_define_method(rb_mInternalObjectWrapper, "internal_object_id", iow_internal_object_id, 0); | ||||
| 
 | ||||
|     Init_object_tracing(rb_mObjSpace); | ||||
| } | ||||
|  |  | |||
							
								
								
									
										8
									
								
								gc.c
									
										
									
									
									
								
							
							
						
						
									
										8
									
								
								gc.c
									
										
									
									
									
								
							|  | @ -3946,6 +3946,12 @@ gc_count_add_each_types(VALUE hash, const char *name, const size_t *types) | |||
| } | ||||
| #endif | ||||
| 
 | ||||
| size_t | ||||
| rb_gc_count(void) | ||||
| { | ||||
|     return rb_objspace.count; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  *  call-seq: | ||||
|  *     GC.count -> Integer | ||||
|  | @ -3959,7 +3965,7 @@ gc_count_add_each_types(VALUE hash, const char *name, const size_t *types) | |||
| static VALUE | ||||
| gc_count(VALUE self) | ||||
| { | ||||
|     return UINT2NUM(rb_objspace.count); | ||||
|     return SIZET2NUM(rb_gc_count()); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
|  | @ -443,6 +443,9 @@ void rb_gc_mark_global_tbl(void); | |||
| void rb_mark_generic_ivar(VALUE); | ||||
| void rb_mark_generic_ivar_tbl(void); | ||||
| 
 | ||||
| /* gc.c */ | ||||
| size_t rb_gc_count(); | ||||
| 
 | ||||
| RUBY_SYMBOL_EXPORT_END | ||||
| 
 | ||||
| #if defined(__cplusplus) | ||||
|  |  | |||
|  | @ -108,4 +108,22 @@ class TestObjSpace < Test::Unit::TestCase | |||
|     } | ||||
|     eom | ||||
|   end | ||||
| 
 | ||||
|   def test_traceobject | ||||
|     o0 = Object.new | ||||
|     ObjectSpace.trace_object_allocations{ | ||||
|       o1 = Object.new; line1 = __LINE__ | ||||
|       o2 = "a"+"b"   ; line2 = __LINE__ | ||||
|       o3 = [1, 2]    ; line3 = __LINE__ | ||||
| 
 | ||||
|       assert_equal(nil, ObjectSpace.allocation_sourcefile(o0)) | ||||
|       assert_equal(nil, ObjectSpace.allocation_sourceline(o0)) | ||||
|       assert_equal(__FILE__, ObjectSpace.allocation_sourcefile(o1)) | ||||
|       assert_equal(line1,    ObjectSpace.allocation_sourceline(o1)) | ||||
|       assert_equal(__FILE__, ObjectSpace.allocation_sourcefile(o2)) | ||||
|       assert_equal(line2,    ObjectSpace.allocation_sourceline(o2)) | ||||
|       assert_equal(__FILE__, ObjectSpace.allocation_sourcefile(o3)) | ||||
|       assert_equal(line3,    ObjectSpace.allocation_sourceline(o3)) | ||||
|     } | ||||
|   end | ||||
| end | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 ko1
						ko1