mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
Prohibit calling undefined allocator [Bug #16297]
This commit is contained in:
parent
5251d18982
commit
f72dc407f2
2 changed files with 37 additions and 2 deletions
31
object.c
31
object.c
|
@ -2106,6 +2106,9 @@ rb_undefined_alloc(VALUE klass)
|
|||
klass);
|
||||
}
|
||||
|
||||
static rb_alloc_func_t class_get_alloc_func(VALUE klass);
|
||||
static VALUE class_call_alloc_func(rb_alloc_func_t allocator, VALUE klass);
|
||||
|
||||
/*
|
||||
* call-seq:
|
||||
* class.allocate() -> obj
|
||||
|
@ -2128,10 +2131,27 @@ rb_undefined_alloc(VALUE klass)
|
|||
*
|
||||
*/
|
||||
|
||||
static VALUE
|
||||
rb_class_alloc_m(VALUE klass)
|
||||
{
|
||||
rb_alloc_func_t allocator = class_get_alloc_func(klass);
|
||||
if (!rb_obj_respond_to(klass, rb_intern("allocate"), 1)) {
|
||||
rb_raise(rb_eTypeError, "calling %"PRIsVALUE".allocate is prohibited",
|
||||
klass);
|
||||
}
|
||||
return class_call_alloc_func(allocator, klass);
|
||||
}
|
||||
|
||||
static VALUE
|
||||
rb_class_alloc(VALUE klass)
|
||||
{
|
||||
VALUE obj;
|
||||
rb_alloc_func_t allocator = class_get_alloc_func(klass);
|
||||
return class_call_alloc_func(allocator, klass);
|
||||
}
|
||||
|
||||
static rb_alloc_func_t
|
||||
class_get_alloc_func(VALUE klass)
|
||||
{
|
||||
rb_alloc_func_t allocator;
|
||||
|
||||
if (RCLASS_SUPER(klass) == 0 && klass != rb_cBasicObject) {
|
||||
|
@ -2144,6 +2164,13 @@ rb_class_alloc(VALUE klass)
|
|||
if (!allocator) {
|
||||
rb_undefined_alloc(klass);
|
||||
}
|
||||
return allocator;
|
||||
}
|
||||
|
||||
static VALUE
|
||||
class_call_alloc_func(rb_alloc_func_t allocator, VALUE klass)
|
||||
{
|
||||
VALUE obj;
|
||||
|
||||
RUBY_DTRACE_CREATE_HOOK(OBJECT, rb_class2name(klass));
|
||||
|
||||
|
@ -4689,7 +4716,7 @@ InitVM_Object(void)
|
|||
rb_define_method(rb_cModule, "deprecate_constant", rb_mod_deprecate_constant, -1); /* in variable.c */
|
||||
rb_define_method(rb_cModule, "singleton_class?", rb_mod_singleton_p, 0);
|
||||
|
||||
rb_define_method(rb_cClass, "allocate", rb_class_alloc, 0);
|
||||
rb_define_method(rb_cClass, "allocate", rb_class_alloc_m, 0);
|
||||
rb_define_method(rb_cClass, "new", rb_class_s_new, -1);
|
||||
rb_define_method(rb_cClass, "initialize", rb_class_initialize, -1);
|
||||
rb_define_method(rb_cClass, "superclass", rb_class_superclass, 0);
|
||||
|
|
|
@ -274,6 +274,14 @@ class TestClass < Test::Unit::TestCase
|
|||
assert_raise(TypeError) { Class.allocate.superclass }
|
||||
bug6863 = '[ruby-core:47148]'
|
||||
assert_raise(TypeError, bug6863) { Class.new(Class.allocate) }
|
||||
|
||||
allocator = Class.instance_method(:allocate)
|
||||
assert_raise_with_message(TypeError, /prohibited/) {
|
||||
allocator.bind(Rational).call
|
||||
}
|
||||
assert_raise_with_message(TypeError, /prohibited/) {
|
||||
allocator.bind_call(Rational)
|
||||
}
|
||||
end
|
||||
|
||||
def test_nonascii_name
|
||||
|
|
Loading…
Reference in a new issue