Skip to content
Projects
Groups
Snippets
Help
Loading...
Sign in / Register
Toggle navigation
P
protobuf
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Packages
Packages
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
submodule
protobuf
Commits
dd45c0b9
Commit
dd45c0b9
authored
Aug 30, 2016
by
Joshua Haberman
Committed by
Jisi Liu
Aug 30, 2016
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Merge pull request #2012 from haberman/rubymapgcfix
Ruby: make sure map parsing frames are GC-rooted.
parent
8335b7d9
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
59 additions
and
9 deletions
+59
-9
encode_decode.c
ruby/ext/google/protobuf_c/encode_decode.c
+54
-8
protobuf.c
ruby/ext/google/protobuf_c/protobuf.c
+2
-0
protobuf.h
ruby/ext/google/protobuf_c/protobuf.h
+2
-0
upb.c
ruby/ext/google/protobuf_c/upb.c
+1
-1
No files found.
ruby/ext/google/protobuf_c/encode_decode.c
View file @
dd45c0b9
...
...
@@ -255,10 +255,54 @@ typedef struct {
// value into the map.
typedef
struct
{
VALUE
map
;
const
map_handlerdata_t
*
handlerdata
;
char
key_storage
[
NATIVE_SLOT_MAX_SIZE
];
char
value_storage
[
NATIVE_SLOT_MAX_SIZE
];
}
map_parse_frame_t
;
static
void
MapParseFrame_mark
(
void
*
_self
)
{
map_parse_frame_t
*
frame
=
_self
;
// This shouldn't strictly be necessary since this should be rooted by the
// message itself, but it can't hurt.
rb_gc_mark
(
frame
->
map
);
native_slot_mark
(
frame
->
handlerdata
->
key_field_type
,
&
frame
->
key_storage
);
native_slot_mark
(
frame
->
handlerdata
->
value_field_type
,
&
frame
->
value_storage
);
}
void
MapParseFrame_free
(
void
*
self
)
{
xfree
(
self
);
}
rb_data_type_t
MapParseFrame_type
=
{
"MapParseFrame"
,
{
MapParseFrame_mark
,
MapParseFrame_free
,
NULL
},
};
// Array of Ruby objects wrapping map_parse_frame_t.
// We don't allow multiple concurrent decodes, so we assume that this global
// variable is specific to the "current" decode.
VALUE
map_parse_frames
;
static
map_parse_frame_t
*
map_push_frame
(
VALUE
map
,
const
map_handlerdata_t
*
handlerdata
)
{
map_parse_frame_t
*
frame
=
ALLOC
(
map_parse_frame_t
);
frame
->
handlerdata
=
handlerdata
;
frame
->
map
=
map
;
native_slot_init
(
handlerdata
->
key_field_type
,
&
frame
->
key_storage
);
native_slot_init
(
handlerdata
->
value_field_type
,
&
frame
->
value_storage
);
rb_ary_push
(
map_parse_frames
,
TypedData_Wrap_Struct
(
rb_cObject
,
&
MapParseFrame_type
,
frame
));
return
frame
;
}
static
void
map_pop_frame
()
{
rb_ary_pop
(
map_parse_frames
);
}
// Handler to begin a map entry: allocates a temporary frame. This is the
// 'startsubmsg' handler on the msgdef that contains the map field.
static
void
*
startmapentry_handler
(
void
*
closure
,
const
void
*
hd
)
{
...
...
@@ -266,13 +310,7 @@ static void *startmapentry_handler(void *closure, const void *hd) {
const
map_handlerdata_t
*
mapdata
=
hd
;
VALUE
map_rb
=
DEREF
(
msg
,
mapdata
->
ofs
,
VALUE
);
map_parse_frame_t
*
frame
=
ALLOC
(
map_parse_frame_t
);
frame
->
map
=
map_rb
;
native_slot_init
(
mapdata
->
key_field_type
,
&
frame
->
key_storage
);
native_slot_init
(
mapdata
->
value_field_type
,
&
frame
->
value_storage
);
return
frame
;
return
map_push_frame
(
map_rb
,
mapdata
);
}
// Handler to end a map entry: inserts the value defined during the message into
...
...
@@ -298,7 +336,7 @@ static bool endmap_handler(void *closure, const void *hd, upb_status* s) {
&
frame
->
value_storage
);
Map_index_set
(
frame
->
map
,
key
,
value
);
xfree
(
frame
);
map_pop_frame
(
);
return
true
;
}
...
...
@@ -737,6 +775,10 @@ VALUE Message_decode(VALUE klass, VALUE data) {
msg_rb
=
rb_class_new_instance
(
0
,
NULL
,
msgklass
);
TypedData_Get_Struct
(
msg_rb
,
MessageHeader
,
&
Message_type
,
msg
);
// We generally expect this to be clear already, but clear it in case parsing
// previously got interrupted somehow.
rb_ary_clear
(
map_parse_frames
);
{
const
upb_pbdecodermethod
*
method
=
msgdef_decodermethod
(
desc
);
const
upb_handlers
*
h
=
upb_pbdecodermethod_desthandlers
(
method
);
...
...
@@ -781,6 +823,10 @@ VALUE Message_decode_json(VALUE klass, VALUE data) {
msg_rb
=
rb_class_new_instance
(
0
,
NULL
,
msgklass
);
TypedData_Get_Struct
(
msg_rb
,
MessageHeader
,
&
Message_type
,
msg
);
// We generally expect this to be clear already, but clear it in case parsing
// previously got interrupted somehow.
rb_ary_clear
(
map_parse_frames
);
{
const
upb_json_parsermethod
*
method
=
msgdef_jsonparsermethod
(
desc
);
stackenv
se
;
...
...
ruby/ext/google/protobuf_c/protobuf.c
View file @
dd45c0b9
...
...
@@ -112,4 +112,6 @@ void Init_protobuf_c() {
upb_def_to_ruby_obj_map
=
rb_hash_new
();
rb_gc_register_address
(
&
upb_def_to_ruby_obj_map
);
map_parse_frames
=
rb_ary_new
();
rb_gc_register_address
(
&
map_parse_frames
);
}
ruby/ext/google/protobuf_c/protobuf.h
View file @
dd45c0b9
...
...
@@ -166,6 +166,8 @@ extern VALUE cBuilder;
extern
VALUE
cError
;
extern
VALUE
cParseError
;
extern
VALUE
map_parse_frames
;
// We forward-declare all of the Ruby method implementations here because we
// sometimes call the methods directly across .c files, rather than going
// through Ruby's method dispatching (e.g. during message parse). It's cleaner
...
...
ruby/ext/google/protobuf_c/upb.c
View file @
dd45c0b9
...
...
@@ -11175,7 +11175,7 @@ static bool parse_mapentry_key(upb_json_parser *p) {
sel
=
getsel_for_handlertype
(
p
,
UPB_HANDLER_STRING
);
upb_sink_putstring
(
&
subsink
,
sel
,
buf
,
len
,
NULL
);
sel
=
getsel_for_handlertype
(
p
,
UPB_HANDLER_ENDSTR
);
upb_sink_endstr
(
&
sub
sink
,
sel
);
upb_sink_endstr
(
&
p
->
top
->
sink
,
sel
);
multipart_end
(
p
);
break
;
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment