binding_of_caller 0.4.0 → 0.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/ext/binding_of_caller/binding_of_caller.c +100 -17
- data/lib/binding_of_caller/version.rb +1 -1
- data/test/test.rb +66 -31
- metadata +30 -34
|
@@ -1,15 +1,31 @@
|
|
|
1
1
|
/* (c) 2011 John Mair (banisterfiend), MIT license */
|
|
2
2
|
|
|
3
3
|
#include <ruby.h>
|
|
4
|
-
|
|
5
|
-
#include <ruby/io.h>
|
|
6
|
-
#include <ruby/re.h>
|
|
7
4
|
#include "vm_core.h"
|
|
8
5
|
#include "gc.h"
|
|
9
6
|
|
|
10
7
|
typedef enum { false, true } bool;
|
|
11
8
|
|
|
12
|
-
|
|
9
|
+
static VALUE
|
|
10
|
+
string2sym(const char * string)
|
|
11
|
+
{
|
|
12
|
+
return ID2SYM(rb_intern(string));
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
static inline const rb_data_type_t *
|
|
16
|
+
threadptr_data_type(void)
|
|
17
|
+
{
|
|
18
|
+
static const rb_data_type_t *thread_data_type;
|
|
19
|
+
if (!thread_data_type) {
|
|
20
|
+
VALUE current_thread = rb_thread_current();
|
|
21
|
+
thread_data_type = RTYPEDDATA_TYPE(current_thread);
|
|
22
|
+
}
|
|
23
|
+
return thread_data_type;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
#define ruby_thread_data_type *threadptr_data_type()
|
|
27
|
+
|
|
28
|
+
#define ruby_current_thread ((rb_thread_t *)RTYPEDDATA_DATA(rb_thread_current()))
|
|
13
29
|
|
|
14
30
|
static size_t
|
|
15
31
|
binding_memsize(const void *ptr)
|
|
@@ -59,31 +75,29 @@ binding_alloc(VALUE klass)
|
|
|
59
75
|
}
|
|
60
76
|
|
|
61
77
|
static bool valid_frame_p(rb_control_frame_t * cfp, rb_control_frame_t * limit_cfp) {
|
|
62
|
-
if (cfp > limit_cfp)
|
|
63
|
-
rb_raise(rb_eRuntimeError, "Invalid frame, gone beyond end of stack!");
|
|
64
|
-
|
|
65
78
|
return cfp->iseq && !NIL_P(cfp->self);
|
|
66
79
|
}
|
|
67
80
|
|
|
68
81
|
static rb_control_frame_t * find_valid_frame(rb_control_frame_t * cfp, rb_control_frame_t * limit_cfp) {
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
while (error_count <= max_frame_errors) {
|
|
82
|
+
while (cfp < limit_cfp) {
|
|
72
83
|
cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp);
|
|
73
84
|
|
|
85
|
+
if (cfp >= limit_cfp)
|
|
86
|
+
return NULL;
|
|
87
|
+
|
|
74
88
|
if (valid_frame_p(cfp, limit_cfp))
|
|
75
89
|
return cfp;
|
|
76
|
-
else
|
|
77
|
-
error_count += 1;
|
|
78
90
|
}
|
|
79
91
|
|
|
80
|
-
//
|
|
81
|
-
return
|
|
92
|
+
// beyond end of stack
|
|
93
|
+
return NULL;
|
|
82
94
|
}
|
|
83
95
|
|
|
84
96
|
static VALUE binding_of_caller(VALUE self, VALUE rb_level)
|
|
85
97
|
{
|
|
86
|
-
rb_thread_t *th
|
|
98
|
+
rb_thread_t *th;
|
|
99
|
+
GetThreadPtr(rb_thread_current(), th);
|
|
100
|
+
|
|
87
101
|
rb_control_frame_t *cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(th->cfp);
|
|
88
102
|
rb_control_frame_t *limit_cfp = (void *)(th->stack + th->stack_size);
|
|
89
103
|
int level = FIX2INT(rb_level);
|
|
@@ -92,6 +106,9 @@ static VALUE binding_of_caller(VALUE self, VALUE rb_level)
|
|
|
92
106
|
for (int i = 0; i < level; i++) {
|
|
93
107
|
cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp);
|
|
94
108
|
|
|
109
|
+
if (cfp >= limit_cfp)
|
|
110
|
+
rb_raise(rb_eRuntimeError, "Invalid frame, gone beyond end of stack!");
|
|
111
|
+
|
|
95
112
|
// skip invalid frames
|
|
96
113
|
if (!valid_frame_p(cfp, limit_cfp))
|
|
97
114
|
cfp = find_valid_frame(cfp, limit_cfp);
|
|
@@ -100,23 +117,89 @@ static VALUE binding_of_caller(VALUE self, VALUE rb_level)
|
|
|
100
117
|
VALUE bindval = binding_alloc(rb_cBinding);
|
|
101
118
|
rb_binding_t *bind;
|
|
102
119
|
|
|
103
|
-
if (cfp == 0)
|
|
120
|
+
if (cfp == 0)
|
|
104
121
|
rb_raise(rb_eRuntimeError, "Can't create Binding Object on top of Fiber.");
|
|
105
|
-
}
|
|
106
122
|
|
|
107
123
|
GetBindingPtr(bindval, bind);
|
|
108
124
|
bind->env = rb_vm_make_env_object(th, cfp);
|
|
109
125
|
bind->filename = cfp->iseq->filename;
|
|
110
126
|
bind->line_no = rb_vm_get_sourceline(cfp);
|
|
127
|
+
|
|
128
|
+
rb_iv_set(bindval, "@frame_type", cfp->flag);
|
|
111
129
|
return bindval;
|
|
112
130
|
}
|
|
113
131
|
|
|
132
|
+
|
|
133
|
+
static VALUE
|
|
134
|
+
frametype_name(VALUE flag)
|
|
135
|
+
{
|
|
136
|
+
switch (flag & VM_FRAME_MAGIC_MASK) {
|
|
137
|
+
case VM_FRAME_MAGIC_METHOD: return string2sym("method");
|
|
138
|
+
case VM_FRAME_MAGIC_BLOCK: return string2sym("block");
|
|
139
|
+
case VM_FRAME_MAGIC_CLASS: return string2sym("class");
|
|
140
|
+
case VM_FRAME_MAGIC_TOP: return string2sym("top");
|
|
141
|
+
case VM_FRAME_MAGIC_FINISH: return string2sym("finish");
|
|
142
|
+
case VM_FRAME_MAGIC_CFUNC: return string2sym("cfunc");
|
|
143
|
+
case VM_FRAME_MAGIC_PROC: return string2sym("proc");
|
|
144
|
+
case VM_FRAME_MAGIC_IFUNC: return string2sym("ifunc");
|
|
145
|
+
case VM_FRAME_MAGIC_EVAL: return string2sym("eval");
|
|
146
|
+
case VM_FRAME_MAGIC_LAMBDA: return string2sym("lambda");
|
|
147
|
+
default:
|
|
148
|
+
rb_raise(rb_eRuntimeError, "frame_type can only be returned for bindings created with Binding#of_caller().");
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
static VALUE
|
|
153
|
+
frame_type(VALUE self)
|
|
154
|
+
{
|
|
155
|
+
return frametype_name(rb_iv_get(self, "@frame_type"));
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
static VALUE frame_count(VALUE self)
|
|
159
|
+
{
|
|
160
|
+
rb_thread_t *th;
|
|
161
|
+
GetThreadPtr(rb_thread_current(), th);
|
|
162
|
+
|
|
163
|
+
rb_control_frame_t *cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(th->cfp);
|
|
164
|
+
rb_control_frame_t *limit_cfp = (void *)(th->stack + th->stack_size);
|
|
165
|
+
|
|
166
|
+
int i = 1;
|
|
167
|
+
while (cfp < limit_cfp) {
|
|
168
|
+
cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp);
|
|
169
|
+
|
|
170
|
+
if (cfp >= limit_cfp)
|
|
171
|
+
return INT2FIX(i);
|
|
172
|
+
|
|
173
|
+
// skip invalid frames
|
|
174
|
+
if (!valid_frame_p(cfp, limit_cfp))
|
|
175
|
+
cfp = find_valid_frame(cfp, limit_cfp);
|
|
176
|
+
|
|
177
|
+
i++;
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
return INT2FIX(i);
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
static VALUE
|
|
184
|
+
callers(VALUE self)
|
|
185
|
+
{
|
|
186
|
+
VALUE ary = rb_ary_new();
|
|
187
|
+
|
|
188
|
+
for (int i = 0; i < FIX2INT(frame_count(self)); i++)
|
|
189
|
+
rb_ary_push(ary, binding_of_caller(self, INT2FIX(i)));
|
|
190
|
+
|
|
191
|
+
return ary;
|
|
192
|
+
}
|
|
193
|
+
|
|
114
194
|
void
|
|
115
195
|
Init_binding_of_caller()
|
|
116
196
|
{
|
|
117
197
|
VALUE mBindingOfCaller = rb_define_module("BindingOfCaller");
|
|
118
198
|
|
|
119
199
|
rb_define_method(mBindingOfCaller, "of_caller", binding_of_caller, 1);
|
|
200
|
+
rb_define_method(mBindingOfCaller, "frame_count", frame_count, 0);
|
|
201
|
+
rb_define_method(mBindingOfCaller, "frame_type", frame_type, 0);
|
|
202
|
+
rb_define_method(mBindingOfCaller, "callers", callers, 0);
|
|
120
203
|
rb_include_module(rb_cBinding, mBindingOfCaller);
|
|
121
204
|
}
|
|
122
205
|
|
data/test/test.rb
CHANGED
|
@@ -8,52 +8,87 @@ puts "Testing binding_of_caller version #{BindingOfCaller::VERSION}..."
|
|
|
8
8
|
puts "Ruby version: #{RUBY_VERSION}"
|
|
9
9
|
|
|
10
10
|
describe BindingOfCaller do
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
11
|
+
describe "of_caller" do
|
|
12
|
+
it "should fetch immediate caller's binding when 0 is passed" do
|
|
13
|
+
o = Object.new
|
|
14
|
+
def o.a
|
|
15
|
+
var = 1
|
|
16
|
+
binding.of_caller(0).eval('var')
|
|
17
|
+
end
|
|
18
|
+
|
|
19
|
+
o. a.should == 1
|
|
16
20
|
end
|
|
17
21
|
|
|
18
|
-
|
|
19
|
-
|
|
22
|
+
it "should fetch parent of caller's binding when 1 is passed" do
|
|
23
|
+
o = Object.new
|
|
24
|
+
def o.a
|
|
25
|
+
var = 1
|
|
26
|
+
b
|
|
27
|
+
end
|
|
20
28
|
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
var = 1
|
|
25
|
-
b
|
|
26
|
-
end
|
|
29
|
+
def o.b
|
|
30
|
+
binding.of_caller(1).eval('var')
|
|
31
|
+
end
|
|
27
32
|
|
|
28
|
-
|
|
29
|
-
binding.of_caller(1).eval('var')
|
|
33
|
+
o.a.should == 1
|
|
30
34
|
end
|
|
31
35
|
|
|
32
|
-
|
|
33
|
-
|
|
36
|
+
it "should modify locals in parent of caller's binding" do
|
|
37
|
+
o = Object.new
|
|
38
|
+
def o.a
|
|
39
|
+
var = 1
|
|
40
|
+
b
|
|
41
|
+
var
|
|
42
|
+
end
|
|
43
|
+
|
|
44
|
+
def o.b
|
|
45
|
+
binding.of_caller(1).eval('var = 20')
|
|
46
|
+
end
|
|
34
47
|
|
|
35
|
-
|
|
36
|
-
o = Object.new
|
|
37
|
-
def o.a
|
|
38
|
-
var = 1
|
|
39
|
-
b
|
|
40
|
-
var
|
|
48
|
+
o.a.should == 20
|
|
41
49
|
end
|
|
42
50
|
|
|
43
|
-
|
|
44
|
-
|
|
51
|
+
it "should raise an exception when retrieving an out of band binding" do
|
|
52
|
+
o = Object.new
|
|
53
|
+
def o.a
|
|
54
|
+
binding.of_caller(100)
|
|
55
|
+
end
|
|
56
|
+
|
|
57
|
+
lambda { o.a }.should.raise RuntimeError
|
|
45
58
|
end
|
|
59
|
+
end
|
|
46
60
|
|
|
47
|
-
|
|
61
|
+
describe "frame_count" do
|
|
62
|
+
it 'frame_count should equal callers.count' do
|
|
63
|
+
binding.frame_count.should == binding.callers.count
|
|
64
|
+
end
|
|
48
65
|
end
|
|
49
66
|
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
67
|
+
describe "frame_type" do
|
|
68
|
+
it 'should return the correct frame types' do
|
|
69
|
+
o = Object.new
|
|
70
|
+
|
|
71
|
+
# method frame
|
|
72
|
+
def o.a
|
|
73
|
+
b
|
|
74
|
+
end
|
|
75
|
+
|
|
76
|
+
# method frame
|
|
77
|
+
def o.b
|
|
78
|
+
# block frame
|
|
79
|
+
proc do
|
|
80
|
+
binding.callers
|
|
81
|
+
end.call
|
|
82
|
+
end
|
|
83
|
+
caller_bindings = o.a
|
|
84
|
+
caller_bindings[0].frame_type.should == :block
|
|
85
|
+
caller_bindings[1].frame_type.should == :method
|
|
86
|
+
caller_bindings[2].frame_type.should == :method
|
|
54
87
|
end
|
|
55
88
|
|
|
56
|
-
|
|
89
|
+
it 'should raise when invoked on an ordinary binding (i.e one not generated through Binding#of_caller)' do
|
|
90
|
+
lambda { binding.frame_type }.should.raise RuntimeError
|
|
91
|
+
end
|
|
57
92
|
end
|
|
58
93
|
end
|
|
59
94
|
|
metadata
CHANGED
|
@@ -1,37 +1,35 @@
|
|
|
1
|
-
--- !ruby/object:Gem::Specification
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
|
2
2
|
name: binding_of_caller
|
|
3
|
-
version: !ruby/object:Gem::Version
|
|
3
|
+
version: !ruby/object:Gem::Version
|
|
4
|
+
version: 0.5.0
|
|
4
5
|
prerelease:
|
|
5
|
-
version: 0.4.0
|
|
6
6
|
platform: ruby
|
|
7
|
-
authors:
|
|
7
|
+
authors:
|
|
8
8
|
- John Mair (banisterfiend)
|
|
9
9
|
autorequire:
|
|
10
10
|
bindir: bin
|
|
11
11
|
cert_chain: []
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
- !ruby/object:Gem::Dependency
|
|
12
|
+
date: 2011-11-05 00:00:00.000000000 Z
|
|
13
|
+
dependencies:
|
|
14
|
+
- !ruby/object:Gem::Dependency
|
|
16
15
|
name: bacon
|
|
17
|
-
|
|
18
|
-
requirement: &id001 !ruby/object:Gem::Requirement
|
|
16
|
+
requirement: &70308421840640 !ruby/object:Gem::Requirement
|
|
19
17
|
none: false
|
|
20
|
-
requirements:
|
|
18
|
+
requirements:
|
|
21
19
|
- - ~>
|
|
22
|
-
- !ruby/object:Gem::Version
|
|
20
|
+
- !ruby/object:Gem::Version
|
|
23
21
|
version: 1.1.0
|
|
24
22
|
type: :development
|
|
25
|
-
|
|
26
|
-
|
|
23
|
+
prerelease: false
|
|
24
|
+
version_requirements: *70308421840640
|
|
25
|
+
description: Retrieve the binding of a method's caller. Can also retrieve bindings
|
|
26
|
+
even further up the stack. Currently only works for MRI 1.9.2.
|
|
27
27
|
email: jrmair@gmail.com
|
|
28
28
|
executables: []
|
|
29
|
-
|
|
30
|
-
extensions:
|
|
29
|
+
extensions:
|
|
31
30
|
- ext/binding_of_caller/extconf.rb
|
|
32
31
|
extra_rdoc_files: []
|
|
33
|
-
|
|
34
|
-
files:
|
|
32
|
+
files:
|
|
35
33
|
- .gemtest
|
|
36
34
|
- .gitignore
|
|
37
35
|
- .yardopts
|
|
@@ -94,30 +92,28 @@ files:
|
|
|
94
92
|
- test/test.rb
|
|
95
93
|
homepage: http://github.com/banister/binding_of_caller
|
|
96
94
|
licenses: []
|
|
97
|
-
|
|
98
95
|
post_install_message:
|
|
99
96
|
rdoc_options: []
|
|
100
|
-
|
|
101
|
-
require_paths:
|
|
97
|
+
require_paths:
|
|
102
98
|
- lib
|
|
103
|
-
required_ruby_version: !ruby/object:Gem::Requirement
|
|
99
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
|
104
100
|
none: false
|
|
105
|
-
requirements:
|
|
106
|
-
- -
|
|
107
|
-
- !ruby/object:Gem::Version
|
|
108
|
-
version:
|
|
109
|
-
required_rubygems_version: !ruby/object:Gem::Requirement
|
|
101
|
+
requirements:
|
|
102
|
+
- - ! '>='
|
|
103
|
+
- !ruby/object:Gem::Version
|
|
104
|
+
version: '0'
|
|
105
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
|
110
106
|
none: false
|
|
111
|
-
requirements:
|
|
112
|
-
- -
|
|
113
|
-
- !ruby/object:Gem::Version
|
|
114
|
-
version:
|
|
107
|
+
requirements:
|
|
108
|
+
- - ! '>='
|
|
109
|
+
- !ruby/object:Gem::Version
|
|
110
|
+
version: '0'
|
|
115
111
|
requirements: []
|
|
116
|
-
|
|
117
112
|
rubyforge_project:
|
|
118
113
|
rubygems_version: 1.8.11
|
|
119
114
|
signing_key:
|
|
120
115
|
specification_version: 3
|
|
121
|
-
summary: Retrieve the binding of a method's caller. Can also retrieve bindings even
|
|
122
|
-
|
|
116
|
+
summary: Retrieve the binding of a method's caller. Can also retrieve bindings even
|
|
117
|
+
further up the stack. Currently only works for MRI 1.9.2.
|
|
118
|
+
test_files:
|
|
123
119
|
- test/test.rb
|