mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
merge revision(s) c9423b016cfeab852bc5a829e55e0a11f80b3ab7,0b1e26398e018116180bf41cb63887f77d5d1b82,78ee2c245331e353e218b8fac9ca722a2bcd8fea: [Backport #15968]
marshal.c: check instance variable count * marshal.c (w_obj_each): ensure that no instance variable was added while dumping other instance variables. [Bug #15968] Hoisted out w_ivar_each marshal.c: check instance variable count * marshal.c (w_ivar_each): ensure that no instance variable was removed while dumping other instance variables. [Bug #15968] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/branches/ruby_2_6@67832 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
00f4c13e22
commit
ec2934d504
3 changed files with 79 additions and 11 deletions
42
marshal.c
42
marshal.c
|
|
@ -257,7 +257,7 @@ class2path(VALUE klass)
|
||||||
}
|
}
|
||||||
|
|
||||||
static void w_long(long, struct dump_arg*);
|
static void w_long(long, struct dump_arg*);
|
||||||
static void w_encoding(VALUE encname, struct dump_call_arg *arg);
|
static int w_encoding(VALUE encname, struct dump_call_arg *arg);
|
||||||
static VALUE encoding_name(VALUE obj, struct dump_arg *arg);
|
static VALUE encoding_name(VALUE obj, struct dump_arg *arg);
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
|
@ -550,14 +550,25 @@ w_uclass(VALUE obj, VALUE super, struct dump_arg *arg)
|
||||||
|
|
||||||
#define to_be_skipped_id(id) (id == rb_id_encoding() || id == rb_intern("E") || !rb_id2str(id))
|
#define to_be_skipped_id(id) (id == rb_id_encoding() || id == rb_intern("E") || !rb_id2str(id))
|
||||||
|
|
||||||
|
struct w_ivar_arg {
|
||||||
|
struct dump_call_arg *dump;
|
||||||
|
st_data_t num_ivar;
|
||||||
|
};
|
||||||
|
|
||||||
static int
|
static int
|
||||||
w_obj_each(st_data_t key, st_data_t val, st_data_t a)
|
w_obj_each(st_data_t key, st_data_t val, st_data_t a)
|
||||||
{
|
{
|
||||||
ID id = (ID)key;
|
ID id = (ID)key;
|
||||||
VALUE value = (VALUE)val;
|
VALUE value = (VALUE)val;
|
||||||
struct dump_call_arg *arg = (struct dump_call_arg *)a;
|
struct w_ivar_arg *ivarg = (struct w_ivar_arg *)a;
|
||||||
|
struct dump_call_arg *arg = ivarg->dump;
|
||||||
|
|
||||||
if (to_be_skipped_id(id)) return ST_CONTINUE;
|
if (to_be_skipped_id(id)) return ST_CONTINUE;
|
||||||
|
if (!ivarg->num_ivar) {
|
||||||
|
rb_raise(rb_eRuntimeError, "instance variable added to %"PRIsVALUE" instance",
|
||||||
|
CLASS_OF(arg->obj));
|
||||||
|
}
|
||||||
|
--ivarg->num_ivar;
|
||||||
w_symbol(ID2SYM(id), arg->arg);
|
w_symbol(ID2SYM(id), arg->arg);
|
||||||
w_object(value, arg->arg, arg->limit);
|
w_object(value, arg->arg, arg->limit);
|
||||||
return ST_CONTINUE;
|
return ST_CONTINUE;
|
||||||
|
|
@ -604,7 +615,7 @@ encoding_name(VALUE obj, struct dump_arg *arg)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static int
|
||||||
w_encoding(VALUE encname, struct dump_call_arg *arg)
|
w_encoding(VALUE encname, struct dump_call_arg *arg)
|
||||||
{
|
{
|
||||||
int limit = arg->limit;
|
int limit = arg->limit;
|
||||||
|
|
@ -614,11 +625,13 @@ w_encoding(VALUE encname, struct dump_call_arg *arg)
|
||||||
case Qtrue:
|
case Qtrue:
|
||||||
w_symbol(ID2SYM(rb_intern("E")), arg->arg);
|
w_symbol(ID2SYM(rb_intern("E")), arg->arg);
|
||||||
w_object(encname, arg->arg, limit);
|
w_object(encname, arg->arg, limit);
|
||||||
|
return 1;
|
||||||
case Qnil:
|
case Qnil:
|
||||||
return;
|
return 0;
|
||||||
}
|
}
|
||||||
w_symbol(ID2SYM(rb_id_encoding()), arg->arg);
|
w_symbol(ID2SYM(rb_id_encoding()), arg->arg);
|
||||||
w_object(encname, arg->arg, limit);
|
w_object(encname, arg->arg, limit);
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static st_index_t
|
static st_index_t
|
||||||
|
|
@ -642,13 +655,25 @@ has_ivars(VALUE obj, VALUE encname, VALUE *ivobj)
|
||||||
return num + enc;
|
return num + enc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
w_ivar_each(VALUE obj, st_index_t num, struct dump_call_arg *arg)
|
||||||
|
{
|
||||||
|
struct w_ivar_arg ivarg = {arg, num};
|
||||||
|
if (!num) return;
|
||||||
|
rb_ivar_foreach(obj, w_obj_each, (st_data_t)&ivarg);
|
||||||
|
if (ivarg.num_ivar) {
|
||||||
|
rb_raise(rb_eRuntimeError, "instance variable removed from %"PRIsVALUE" instance",
|
||||||
|
CLASS_OF(arg->obj));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
w_ivar(st_index_t num, VALUE ivobj, VALUE encname, struct dump_call_arg *arg)
|
w_ivar(st_index_t num, VALUE ivobj, VALUE encname, struct dump_call_arg *arg)
|
||||||
{
|
{
|
||||||
w_long(num, arg->arg);
|
w_long(num, arg->arg);
|
||||||
w_encoding(encname, arg);
|
num -= w_encoding(encname, arg);
|
||||||
if (ivobj != Qundef) {
|
if (ivobj != Qundef) {
|
||||||
rb_ivar_foreach(ivobj, w_obj_each, (st_data_t)arg);
|
w_ivar_each(ivobj, num, arg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -659,9 +684,7 @@ w_objivar(VALUE obj, struct dump_call_arg *arg)
|
||||||
|
|
||||||
rb_ivar_foreach(obj, obj_count_ivars, (st_data_t)&num);
|
rb_ivar_foreach(obj, obj_count_ivars, (st_data_t)&num);
|
||||||
w_long(num, arg->arg);
|
w_long(num, arg->arg);
|
||||||
if (num != 0) {
|
w_ivar_each(obj, num, arg);
|
||||||
rb_ivar_foreach(obj, w_obj_each, (st_data_t)arg);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
|
@ -680,6 +703,7 @@ w_object(VALUE obj, struct dump_arg *arg, int limit)
|
||||||
if (limit > 0) limit--;
|
if (limit > 0) limit--;
|
||||||
c_arg.limit = limit;
|
c_arg.limit = limit;
|
||||||
c_arg.arg = arg;
|
c_arg.arg = arg;
|
||||||
|
c_arg.obj = obj;
|
||||||
|
|
||||||
if (st_lookup(arg->data, obj, &num)) {
|
if (st_lookup(arg->data, obj, &num)) {
|
||||||
w_byte(TYPE_LINK, arg);
|
w_byte(TYPE_LINK, arg);
|
||||||
|
|
|
||||||
|
|
@ -779,4 +779,48 @@ class TestMarshal < Test::Unit::TestCase
|
||||||
obj = Bug14314.new(foo: 42)
|
obj = Bug14314.new(foo: 42)
|
||||||
assert_equal obj, Marshal.load(Marshal.dump(obj))
|
assert_equal obj, Marshal.load(Marshal.dump(obj))
|
||||||
end
|
end
|
||||||
|
|
||||||
|
class Bug15968
|
||||||
|
attr_accessor :bar, :baz
|
||||||
|
|
||||||
|
def initialize
|
||||||
|
self.bar = Bar.new(self)
|
||||||
|
end
|
||||||
|
|
||||||
|
class Bar
|
||||||
|
attr_accessor :foo
|
||||||
|
|
||||||
|
def initialize(foo)
|
||||||
|
self.foo = foo
|
||||||
|
end
|
||||||
|
|
||||||
|
def marshal_dump
|
||||||
|
if self.foo.baz
|
||||||
|
self.foo.remove_instance_variable(:@baz)
|
||||||
|
else
|
||||||
|
self.foo.baz = :problem
|
||||||
|
end
|
||||||
|
{foo: self.foo}
|
||||||
|
end
|
||||||
|
|
||||||
|
def marshal_load(data)
|
||||||
|
self.foo = data[:foo]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_marshal_dump_adding_instance_variable
|
||||||
|
obj = Bug15968.new
|
||||||
|
assert_raise_with_message(RuntimeError, /instance variable added/) do
|
||||||
|
Marshal.dump(obj)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_marshal_dump_removing_instance_variable
|
||||||
|
obj = Bug15968.new
|
||||||
|
obj.baz = :Bug15968
|
||||||
|
assert_raise_with_message(RuntimeError, /instance variable removed/) do
|
||||||
|
Marshal.dump(obj)
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
||||||
|
|
@ -1,10 +1,10 @@
|
||||||
#define RUBY_VERSION "2.6.6"
|
#define RUBY_VERSION "2.6.6"
|
||||||
#define RUBY_RELEASE_DATE RUBY_RELEASE_YEAR_STR"-"RUBY_RELEASE_MONTH_STR"-"RUBY_RELEASE_DAY_STR
|
#define RUBY_RELEASE_DATE RUBY_RELEASE_YEAR_STR"-"RUBY_RELEASE_MONTH_STR"-"RUBY_RELEASE_DAY_STR
|
||||||
#define RUBY_PATCHLEVEL 119
|
#define RUBY_PATCHLEVEL 120
|
||||||
|
|
||||||
#define RUBY_RELEASE_YEAR 2019
|
#define RUBY_RELEASE_YEAR 2019
|
||||||
#define RUBY_RELEASE_MONTH 12
|
#define RUBY_RELEASE_MONTH 12
|
||||||
#define RUBY_RELEASE_DAY 9
|
#define RUBY_RELEASE_DAY 15
|
||||||
|
|
||||||
#include "ruby/version.h"
|
#include "ruby/version.h"
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue