Feature #16375
closedRight size regular expression compile buffers for literal regexes and on Regexp#freeze
Description
References PR https://github.com/ruby/ruby/pull/2696
As a continuation of type specific resize on freeze implementations of String and Array and looking into the Regexp
type I found these memory access patterns for regular expression literals:
==22079== -------------------- 12 of 500 --------------------
==22079== max-live: 1,946,560 in 4,345 blocks
==22079== tot-alloc: 1,946,560 in 4,345 blocks (avg size 448.00)
==22079== deaths: none (none of these blocks were freed)
==22079== acc-ratios: 1.36 rd, 0.98 wr (2,651,994 b-read, 1,908,158 b-written)
==22079== at 0x4C2DECF: malloc (in /usr/lib/valgrind/vgpreload_exp-dhat-amd64-linux.so)
==22079== by 0x24C496: onig_new_with_source (re.c:844)
==22079== by 0x24C496: make_regexp (re.c:874)
==22079== by 0x24C496: rb_reg_initialize (re.c:2858)
==22079== by 0x24C496: rb_reg_initialize_str (re.c:2892)
==22079== by 0x24C496: rb_reg_compile (re.c:2982)
==22079== by 0x12EB84: rb_parser_reg_compile (parse.y:12185)
==22079== by 0x12EB84: parser_reg_compile (parse.y:12179)
==22079== by 0x12EB84: reg_compile (parse.y:12195)
==22079== by 0x2147E3: new_regexp (parse.y:10101)
==22079== by 0x2147E3: ruby_yyparse (parse.y:4419)
==22079== by 0x2161F7: yycompile0 (parse.y:5942)
==22079== by 0x3241FF: rb_suppress_tracing (vm_trace.c:427)
==22079== by 0x1FDBF6: yycompile (parse.y:5991)
==22079== by 0x1FDBF6: rb_parser_compile_file_path (parse.y:6130)
==22079== by 0x27AC96: load_file_internal (ruby.c:2034)
==22079== by 0x137730: rb_ensure (eval.c:1129)
==22079== by 0x27CEEA: load_file (ruby.c:2153)
==22079== by 0x27CEEA: rb_parser_load_file (ruby.c:2175)
==22079== by 0x1954CE: load_iseq_eval (load.c:587)
==22079== by 0x1954CE: rb_load_internal (load.c:651)
==22079== by 0x1954CE: rb_f_load (load.c:709)
==22079== by 0x2FB957: vm_call_cfunc_with_frame (vm_insnhelper.c:2468)
==22079== by 0x2FB957: vm_call_cfunc (vm_insnhelper.c:2493)
Digging a little further and remembering some context of previous oniguruma memory investigation I remembered the pattern buffer struct has a compile buffer with a simple watermark for tracking used space. This changeset implements reg_resize
(static as ary_resize
) which attempts to right size the compile buffer if over allocated at the following sites:
- After compiling a literal regular expression.
- Implement an explicit type specific
rb_reg_freeze
and pointRegexp#compile
to it - I also follow the
chain
member which points to anotherregex_t
on the struct if present, but have not been able to find references to it in the source tree other than for freeing a regex or inspecting it's memory footprint.
I introduced 2 new debug counters, which yields the following results on booting Redmine on Rails 5:
[RUBY_DEBUG_COUNTER] obj_regexp_lit_extracapa 6319
[RUBY_DEBUG_COUNTER] obj_regexp_lit_extracapa_bytes 301685
About 300kb reallocated across 6319 oversized instances.
An example of Regexp#freeze
irb(main):007:0> r = Regexp.compile("(?!%\h\h|[!$-&(-;=?-_a-~]).")
irb(main):008:0> ObjectSpace.memsize_of(r)
=> 588
irb(main):009:0> r.freeze
=> /(?!%hh|[!$-&(-;=?-_a-~])./
irb(main):010:0> ObjectSpace.memsize_of(r)
=> 543
There is likely more layers that can be peeled back here, but keeping it simple and concise for review.
I think it's possible to get towards a state where 5 to 10MB RSS can be shaved off a standard Rails process by:
- Being careful with buffer defaults for objects that have auxiliary buffers
- Identifying hooks where excess allocation can be reduced to current watermark if the object buffer does not need to grow anymore (literal object, frozen object etc.)
- A GC hook on promotion to OLD object to trim excessive capacity, which I'd consider as a special kind of garbage, as outlined in https://bugs.ruby-lang.org/issues/15402
Would any further incremental work towards this be considered worthwhile from a ruby-core and community perspective?
Updated by shevegen (Robert A. Heiler) about 5 years ago
Would any further incremental work towards this be considered worthwhile from a
ruby-core and community perspective?
I can not answer for the ruby-core team, or anyone else, but matz has the motto of
3x3 for ruby 3.0 which is destined for next year.
Some time ago, I think, matz said that with mjit the goal has been more or less
achieved - but matz also likes to say that nobody minds more speed improvement,
even if they are only little optcarrot improvements to get the duck going (I
think it's a duck wanting to get to the carrot). So that would leave lots of
time for next year still, even with small improvements - might be even a fun
competition to get to the goal without mjit. :)
I guess the core team may have a look at any potential side effects or trade
offs of any changes.
Updated by methodmissing (Lourens Naudé) about 5 years ago
shevegen (Robert A. Heiler) wrote:
Would any further incremental work towards this be considered worthwhile from a
ruby-core and community perspective?I can not answer for the ruby-core team, or anyone else, but matz has the motto of
3x3 for ruby 3.0 which is destined for next year.Some time ago, I think, matz said that with mjit the goal has been more or less
achieved - but matz also likes to say that nobody minds more speed improvement,
even if they are only little optcarrot improvements to get the duck going (I
think it's a duck wanting to get to the carrot). So that would leave lots of
time for next year still, even with small improvements - might be even a fun
competition to get to the goal without mjit. :)I guess the core team may have a look at any potential side effects or trade
offs of any changes.
Yes this isn't focused on speed, but more so on being careful with auxiliary memory buffers and inferring where to reduce over allocated ones (especially on immutable or old objects). I expect to never post to the list with an order of magnitude improvement, but do think incrementally scavenging a few kilobytes here with goals of
- Not complicating MRI
- Backwards compatible and non breaking changes
- Not negatively affecting performance in order to reduce memory overhead (usually 1 or the other, rarely can have both)
eventually can add up in the low single digit or even MB reduction range for Rails processes.
After a brief exchange with Jean Boussier (byroot), he noted that regular expressions can be considered immutable when allocated and initialized and as such I dropped the Regexp#freeze
API addition and instead applied the resize optimization to every allocation callsite (4 of them) and got a 50% improvement in compile buffer cleanup by effectively extending the net to a few more regex objects (300kb -> 469kb):
[RUBY_DEBUG_COUNTER] obj_regexp_lit_extracapa 7249
[RUBY_DEBUG_COUNTER] obj_regexp_lit_extracapa_bytes 481204
References https://github.com/ruby/ruby/pull/2696#issuecomment-559147405
Updated by mame (Yusuke Endoh) about 5 years ago
- Tracker changed from Misc to Feature
- Assignee set to nobu (Nobuyoshi Nakada)
- Target version set to 36
@nobu (Nobuyoshi Nakada) will review the patch.
@methodmissing Could you file a ticket about performance improvement and/so refactoring into the Feature tracker?
Updated by methodmissing (Lourens Naudé) about 5 years ago
mame (Yusuke Endoh) wrote:
@nobu (Nobuyoshi Nakada) will review the patch.
@methodmissing Could you file a ticket about performance improvement and/so refactoring into the Feature tracker?
Thanks for the consideration and discussing the proposal. Do you mean file a refactored version of this ticket which has drifted in scope on the Feature tracker with the memory improvements?
Updated by mame (Yusuke Endoh) about 5 years ago
methodmissing (Lourens Naudé) wrote:
Thanks for the consideration and discussing the proposal. Do you mean file a refactored version of this ticket which has drifted in scope on the Feature tracker with the memory improvements?
Sorry, I meant please choose "Feature" (not "Misc") as a tracker if you register a ticket about improvements from the next time.
I've already moved this ticket to Feature tracker. You have to do nothing about this ticket unless nobu requests.
Updated by hsbt (Hiroshi SHIBATA) over 4 years ago
- Target version changed from 36 to 3.0
Updated by mame (Yusuke Endoh) about 4 years ago
- Status changed from Open to Closed
Merged by nobu at 9a17437558e42aa1da372b515ba8bc18067d578c