summaryrefslogtreecommitdiff
path: root/test/-ext-/debug/test_debug.rb
blob: 22298598013d981f1b1211da478fff2a504415b3 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
# frozen_string_literal: false
require 'test/unit'
require '-test-/debug'

class TestDebug < Test::Unit::TestCase

  def binds_check(binds, msg = nil)
    count = Hash.new(0)
    assert_instance_of(Array, binds, msg)
    binds.each{|(_self, bind, klass, iseq, loc)|
      if _self == self
        count[:self] += 1
      end

      if bind
        assert_instance_of(Binding, bind, msg)
        count[:bind] += 1
      end

      if klass
        assert(klass.instance_of?(Module) || klass.instance_of?(Class), msg)
        count[:class] += 1
      end

      if iseq
        count[:iseq] += 1
        assert_instance_of(RubyVM::InstructionSequence, iseq, msg)

        # Backtraces and source locations don't match for :c_trace methods
        unless iseq.disasm.include?('C_TRACE')
          # check same location
          assert_equal(loc.path, iseq.path, msg)
          assert_equal(loc.absolute_path, iseq.absolute_path, msg)
          #assert_equal(loc.label, iseq.label, msg)
          assert_operator(loc.lineno, :>=, iseq.first_lineno, msg)
        end
      end

      assert_instance_of(Thread::Backtrace::Location, loc, msg)

    }
    assert_operator(0, :<, count[:self], msg)
    assert_operator(0, :<, count[:bind], msg)
    assert_operator(0, :<, count[:iseq], msg)
    assert_operator(0, :<, count[:class], msg)
  end

  def test_inspector_open
    binds = Bug::Debug.inspector
    binds_check binds
  end

  def inspector_in_eval
    eval("Bug::Debug.inspector")
  end

  def test_inspector_open_in_eval
    bug7635 = '[ruby-core:51640]'
    binds = inspector_in_eval
    binds_check binds, bug7635
  end

  class MyRelation
    include Enumerable

    def each
      yield :each_entry
    end
  end

  def test_lazy_block
    x = MyRelation.new.any? do
      Bug::Debug.inspector
      true
    end
    assert_equal true, x, '[Bug #15105]'
  end

  # This is a YJIT test, but we can't test this without a C extension that calls
  # rb_debug_inspector_open(), so we're testing it using "-test-/debug" here.
  def test_yjit_invalidates_setlocal_after_inspector_call
    val = setlocal_after_proc_call(proc { Bug::Debug.inspector; :ok })
    assert_equal :ok, val
  end if defined?(RubyVM::YJIT) && RubyVM::YJIT.enabled?

  private

  def setlocal_after_proc_call(block)
    local = block.call # setlocal followed by OPTIMIZED_METHOD_TYPE_CALL
    itself # split a block using a C call
    local # getlocal
  end
end