summaryrefslogtreecommitdiff
path: root/marshal.c
diff options
context:
space:
mode:
authorNobuyoshi Nakada <nobu@ruby-lang.org>2019-07-01 16:20:03 +0900
committerNobuyoshi Nakada <nobu@ruby-lang.org>2019-07-01 16:20:03 +0900
commit78ee2c245331e353e218b8fac9ca722a2bcd8fea (patch)
treee1e3ce2747384d7c422c26cd48c2a380bc3cf939 /marshal.c
parent0b1e26398e018116180bf41cb63887f77d5d1b82 (diff)
downloadruby-78ee2c245331e353e218b8fac9ca722a2bcd8fea.tar.gz
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]
Diffstat (limited to 'marshal.c')
-rw-r--r--marshal.c14
1 files changed, 10 insertions, 4 deletions
diff --git a/marshal.c b/marshal.c
index c89ef962ee..9ac88471c6 100644
--- a/marshal.c
+++ b/marshal.c
@@ -258,7 +258,7 @@ class2path(VALUE klass)
int ruby_marshal_write_long(long x, char *buf);
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 void
@@ -626,7 +626,7 @@ encoding_name(VALUE obj, struct dump_arg *arg)
}
}
-static void
+static int
w_encoding(VALUE encname, struct dump_call_arg *arg)
{
int limit = arg->limit;
@@ -636,11 +636,13 @@ w_encoding(VALUE encname, struct dump_call_arg *arg)
case Qtrue:
w_symbol(ID2SYM(rb_intern("E")), arg->arg);
w_object(encname, arg->arg, limit);
+ return 1;
case Qnil:
- return;
+ return 0;
}
w_symbol(ID2SYM(rb_id_encoding()), arg->arg);
w_object(encname, arg->arg, limit);
+ return 1;
}
static st_index_t
@@ -670,13 +672,17 @@ 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
w_ivar(st_index_t num, VALUE ivobj, VALUE encname, struct dump_call_arg *arg)
{
w_long(num, arg->arg);
- w_encoding(encname, arg);
+ num -= w_encoding(encname, arg);
if (ivobj != Qundef) {
w_ivar_each(ivobj, num, arg);
}