2015-12-16 05:07:31 +00:00
|
|
|
# frozen_string_literal: false
|
2006-03-04 06:28:51 +00:00
|
|
|
require 'test/unit'
|
|
|
|
|
|
|
|
class TestObjectSpace < Test::Unit::TestCase
|
|
|
|
def self.deftest_id2ref(obj)
|
|
|
|
/:(\d+)/ =~ caller[0]
|
|
|
|
file = $`
|
|
|
|
line = $1.to_i
|
2006-03-04 06:39:25 +00:00
|
|
|
code = <<"End"
|
2006-03-04 06:28:51 +00:00
|
|
|
define_method("test_id2ref_#{line}") {\
|
|
|
|
o = ObjectSpace._id2ref(obj.object_id);\
|
2006-09-16 15:30:49 +00:00
|
|
|
assert_same(obj, o, "didn't round trip: \#{obj.inspect}");\
|
2006-03-04 06:28:51 +00:00
|
|
|
}
|
|
|
|
End
|
2006-03-04 06:39:25 +00:00
|
|
|
eval code, binding, file, line
|
2006-03-04 06:28:51 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
deftest_id2ref(-0x4000000000000001)
|
|
|
|
deftest_id2ref(-0x4000000000000000)
|
|
|
|
deftest_id2ref(-0x40000001)
|
|
|
|
deftest_id2ref(-0x40000000)
|
|
|
|
deftest_id2ref(-1)
|
|
|
|
deftest_id2ref(0)
|
|
|
|
deftest_id2ref(1)
|
|
|
|
deftest_id2ref(0x3fffffff)
|
|
|
|
deftest_id2ref(0x40000000)
|
|
|
|
deftest_id2ref(0x3fffffffffffffff)
|
|
|
|
deftest_id2ref(0x4000000000000000)
|
|
|
|
deftest_id2ref(:a)
|
|
|
|
deftest_id2ref(:abcdefghijilkjl)
|
|
|
|
deftest_id2ref(:==)
|
|
|
|
deftest_id2ref(Object.new)
|
|
|
|
deftest_id2ref(self)
|
|
|
|
deftest_id2ref(true)
|
|
|
|
deftest_id2ref(false)
|
|
|
|
deftest_id2ref(nil)
|
2008-04-14 14:26:51 +00:00
|
|
|
|
2019-12-23 15:02:14 +09:00
|
|
|
def test_id2ref_liveness
|
|
|
|
assert_normal_exit <<-EOS
|
|
|
|
ids = []
|
|
|
|
10.times{
|
|
|
|
1_000.times{
|
|
|
|
ids << 'hello'.object_id
|
|
|
|
}
|
|
|
|
objs = ids.map{|id|
|
|
|
|
begin
|
|
|
|
ObjectSpace._id2ref(id)
|
|
|
|
rescue RangeError
|
|
|
|
nil
|
|
|
|
end
|
|
|
|
}
|
|
|
|
GC.start
|
|
|
|
objs.each{|e| e.inspect}
|
|
|
|
}
|
|
|
|
EOS
|
|
|
|
end
|
|
|
|
|
2020-06-16 01:03:15 +09:00
|
|
|
def test_id2ref_invalid_argument
|
|
|
|
msg = /no implicit conversion/
|
|
|
|
assert_raise_with_message(TypeError, msg) {ObjectSpace._id2ref(nil)}
|
|
|
|
assert_raise_with_message(TypeError, msg) {ObjectSpace._id2ref(false)}
|
|
|
|
assert_raise_with_message(TypeError, msg) {ObjectSpace._id2ref(true)}
|
|
|
|
assert_raise_with_message(TypeError, msg) {ObjectSpace._id2ref(:a)}
|
|
|
|
assert_raise_with_message(TypeError, msg) {ObjectSpace._id2ref("0")}
|
|
|
|
assert_raise_with_message(TypeError, msg) {ObjectSpace._id2ref(Object.new)}
|
|
|
|
end
|
|
|
|
|
2008-04-14 14:26:51 +00:00
|
|
|
def test_count_objects
|
|
|
|
h = {}
|
|
|
|
ObjectSpace.count_objects(h)
|
|
|
|
assert_kind_of(Hash, h)
|
2013-12-13 09:18:05 +00:00
|
|
|
assert_empty(h.keys.delete_if {|x| x.is_a?(Symbol) || x.is_a?(Integer) })
|
|
|
|
assert_empty(h.values.delete_if {|x| x.is_a?(Integer) })
|
2008-04-14 14:26:51 +00:00
|
|
|
|
|
|
|
h = ObjectSpace.count_objects
|
|
|
|
assert_kind_of(Hash, h)
|
2013-12-13 09:18:05 +00:00
|
|
|
assert_empty(h.keys.delete_if {|x| x.is_a?(Symbol) || x.is_a?(Integer) })
|
|
|
|
assert_empty(h.values.delete_if {|x| x.is_a?(Integer) })
|
2008-04-14 14:26:51 +00:00
|
|
|
|
|
|
|
assert_raise(TypeError) { ObjectSpace.count_objects(1) }
|
2008-05-31 14:03:23 +00:00
|
|
|
|
|
|
|
h0 = {:T_FOO=>1000}
|
|
|
|
h = ObjectSpace.count_objects(h0)
|
|
|
|
assert_same(h0, h)
|
|
|
|
assert_equal(0, h0[:T_FOO])
|
2008-04-14 14:26:51 +00:00
|
|
|
end
|
2008-06-05 14:57:05 +00:00
|
|
|
|
|
|
|
def test_finalizer
|
2019-07-03 14:44:20 +09:00
|
|
|
assert_in_out_err(["-e", <<-END], "", %w(:ok :ok :ok :ok), [])
|
2008-06-05 14:57:05 +00:00
|
|
|
a = []
|
|
|
|
ObjectSpace.define_finalizer(a) { p :ok }
|
|
|
|
b = a.dup
|
|
|
|
ObjectSpace.define_finalizer(a) { p :ok }
|
2010-06-23 05:32:46 +00:00
|
|
|
!b
|
2008-06-05 14:57:05 +00:00
|
|
|
END
|
2008-07-15 15:26:04 +00:00
|
|
|
assert_raise(ArgumentError) { ObjectSpace.define_finalizer([], Object.new) }
|
2013-11-09 18:12:39 +00:00
|
|
|
|
|
|
|
code = proc do |priv|
|
|
|
|
<<-"CODE"
|
|
|
|
fin = Object.new
|
|
|
|
class << fin
|
|
|
|
#{priv}def call(id)
|
|
|
|
puts "finalized"
|
2013-11-09 15:36:46 +00:00
|
|
|
end
|
2013-11-09 18:12:39 +00:00
|
|
|
end
|
|
|
|
ObjectSpace.define_finalizer([], fin)
|
|
|
|
CODE
|
2013-11-09 15:36:46 +00:00
|
|
|
end
|
2019-07-03 14:44:20 +09:00
|
|
|
assert_in_out_err([], code[""], ["finalized"])
|
|
|
|
assert_in_out_err([], code["private "], ["finalized"])
|
2014-09-10 08:22:03 +00:00
|
|
|
c = EnvUtil.labeled_class("C\u{3042}").new
|
|
|
|
o = Object.new
|
|
|
|
assert_raise_with_message(ArgumentError, /C\u{3042}/) {
|
|
|
|
ObjectSpace.define_finalizer(o, c)
|
|
|
|
}
|
2008-06-05 14:57:05 +00:00
|
|
|
end
|
2012-12-03 10:03:25 +00:00
|
|
|
|
2016-06-20 04:37:19 +00:00
|
|
|
def test_finalizer_with_super
|
|
|
|
assert_in_out_err(["-e", <<-END], "", %w(:ok), [])
|
|
|
|
class A
|
|
|
|
def foo
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
class B < A
|
|
|
|
def foo
|
|
|
|
1.times { super }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
class C
|
|
|
|
module M
|
|
|
|
end
|
|
|
|
|
|
|
|
FINALIZER = proc do
|
2016-08-09 07:36:14 +00:00
|
|
|
M.module_eval(__FILE__, "", __LINE__) do
|
2016-06-20 04:37:19 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def define_finalizer
|
|
|
|
ObjectSpace.define_finalizer(self, FINALIZER)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
class D
|
|
|
|
def foo
|
|
|
|
B.new.foo
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
C::M.singleton_class.send :define_method, :module_eval do |src, id, line|
|
|
|
|
end
|
|
|
|
|
|
|
|
GC.stress = true
|
|
|
|
10.times do
|
|
|
|
C.new.define_finalizer
|
|
|
|
D.new.foo
|
|
|
|
end
|
|
|
|
|
|
|
|
p :ok
|
|
|
|
END
|
|
|
|
end
|
|
|
|
|
2021-07-23 00:32:09 +09:00
|
|
|
def test_exception_in_finalizer
|
|
|
|
assert_in_out_err([], "#{<<~"begin;"}\n#{<<~'end;'}", [], /finalizing \(RuntimeError\)/)
|
|
|
|
begin;
|
|
|
|
ObjectSpace.define_finalizer(Object.new) {raise "finalizing"}
|
|
|
|
end;
|
|
|
|
end
|
|
|
|
|
2021-04-07 11:32:30 -07:00
|
|
|
def test_finalizer_thread_raise
|
|
|
|
GC.disable
|
|
|
|
fzer = proc do |id|
|
|
|
|
sleep 0.2
|
|
|
|
end
|
|
|
|
2.times do
|
|
|
|
o = Object.new
|
|
|
|
ObjectSpace.define_finalizer(o, fzer)
|
|
|
|
end
|
|
|
|
|
|
|
|
my_error = Class.new(RuntimeError)
|
|
|
|
begin
|
|
|
|
main_th = Thread.current
|
|
|
|
Thread.new do
|
|
|
|
sleep 0.1
|
|
|
|
main_th.raise(my_error)
|
|
|
|
end
|
|
|
|
GC.start
|
|
|
|
puts "After GC"
|
|
|
|
sleep(10)
|
|
|
|
assert(false)
|
|
|
|
rescue my_error
|
|
|
|
end
|
2021-08-04 15:33:47 -04:00
|
|
|
ensure
|
|
|
|
GC.enable
|
2021-04-07 11:32:30 -07:00
|
|
|
end
|
|
|
|
|
2012-12-03 10:03:25 +00:00
|
|
|
def test_each_object
|
2015-11-25 03:18:13 +00:00
|
|
|
klass = Class.new
|
|
|
|
new_obj = klass.new
|
|
|
|
|
|
|
|
found = []
|
|
|
|
count = ObjectSpace.each_object(klass) do |obj|
|
|
|
|
found << obj
|
|
|
|
end
|
|
|
|
assert_equal(1, count)
|
|
|
|
assert_equal(1, found.size)
|
|
|
|
assert_same(new_obj, found[0])
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_each_object_enumerator
|
|
|
|
klass = Class.new
|
|
|
|
new_obj = klass.new
|
|
|
|
|
|
|
|
found = []
|
|
|
|
counter = ObjectSpace.each_object(klass)
|
|
|
|
assert_equal(1, counter.each {|obj| found << obj})
|
|
|
|
assert_equal(1, found.size)
|
|
|
|
assert_same(new_obj, found[0])
|
|
|
|
end
|
|
|
|
|
2021-11-02 18:29:53 +09:00
|
|
|
def test_each_object_no_garbage
|
2013-05-20 02:22:35 +00:00
|
|
|
assert_separately([], <<-End)
|
2012-12-03 10:03:25 +00:00
|
|
|
GC.disable
|
|
|
|
eval('begin; 1.times{}; rescue; ensure; end')
|
|
|
|
arys = []
|
|
|
|
ObjectSpace.each_object(Array){|ary|
|
|
|
|
arys << ary
|
|
|
|
}
|
|
|
|
GC.enable
|
|
|
|
arys.each{|ary|
|
2012-12-03 11:06:21 +00:00
|
|
|
begin
|
|
|
|
assert_equal(String, ary.inspect.class) # should not cause SEGV
|
|
|
|
rescue RuntimeError
|
|
|
|
# rescue "can't modify frozen File" error.
|
|
|
|
end
|
2012-12-03 10:03:25 +00:00
|
|
|
}
|
2013-05-20 02:22:35 +00:00
|
|
|
End
|
2012-12-03 10:03:25 +00:00
|
|
|
end
|
2014-12-09 01:16:27 +00:00
|
|
|
|
|
|
|
def test_each_object_recursive_key
|
|
|
|
assert_normal_exit(<<-'end;', '[ruby-core:66742] [Bug #10579]')
|
|
|
|
h = {["foo"]=>nil}
|
|
|
|
p Thread.current[:__recursive_key__]
|
|
|
|
end;
|
|
|
|
end
|
2015-07-21 19:04:56 +00:00
|
|
|
|
|
|
|
def test_each_object_singleton_class
|
|
|
|
assert_separately([], <<-End)
|
|
|
|
class C
|
|
|
|
class << self
|
|
|
|
$c = self
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
exist = false
|
|
|
|
ObjectSpace.each_object(Class){|o|
|
|
|
|
exist = true if $c == o
|
|
|
|
}
|
|
|
|
assert(exist, 'Bug #11360')
|
|
|
|
End
|
2015-12-22 13:15:58 +00:00
|
|
|
|
|
|
|
klass = Class.new
|
|
|
|
instance = klass.new
|
|
|
|
sclass = instance.singleton_class
|
|
|
|
meta = klass.singleton_class
|
|
|
|
assert_kind_of(meta, sclass)
|
|
|
|
assert_include(ObjectSpace.each_object(meta).to_a, sclass)
|
2015-07-21 19:04:56 +00:00
|
|
|
end
|
2021-03-12 19:36:58 +00:00
|
|
|
|
|
|
|
def test_each_object_with_allocation
|
|
|
|
assert_normal_exit(<<-End)
|
|
|
|
list = []
|
|
|
|
ObjectSpace.each_object { |o| list << Object.new }
|
|
|
|
End
|
|
|
|
end
|
2006-03-04 06:28:51 +00:00
|
|
|
end
|