ruby-changes:45308
From: nobu <ko1@a...>
Date: Fri, 20 Jan 2017 13:28:01 +0900 (JST)
Subject: [ruby-changes:45308] nobu:r57381 (trunk): compile.c: optimization of defined? in condition
nobu 2017-01-20 13:27:55 +0900 (Fri, 20 Jan 2017) New Revision: 57381 https://svn.ruby-lang.org/cgi-bin/viewvc.cgi?view=revision&revision=57381 Log: compile.c: optimization of defined? in condition * compile.c (compile_branch_condition): trivial optimization of defined? expression in a branch condition, where a string is not needed, but just a boolean. Modified files: trunk/compile.c Index: compile.c =================================================================== --- compile.c (revision 57380) +++ compile.c (revision 57381) @@ -462,6 +462,8 @@ static int iseq_set_sequence(rb_iseq_t * https://github.com/ruby/ruby/blob/trunk/compile.c#L462 static int iseq_set_exception_table(rb_iseq_t *iseq); static int iseq_set_optargs_table(rb_iseq_t *iseq); +static int compile_defined_expr(rb_iseq_t *iseq, LINK_ANCHOR *const ret, NODE *node, VALUE needstr); + /* * To make Array to LinkedList, use link_anchor */ @@ -2903,8 +2905,12 @@ compile_branch_condition(rb_iseq_t *iseq https://github.com/ruby/ruby/blob/trunk/compile.c#L2905 case NODE_FLIP3: CHECK(compile_flip_flop(iseq, ret, cond, FALSE, then_label, else_label)); break; + case NODE_DEFINED: + CHECK(compile_defined_expr(iseq, ret, cond, Qfalse)); + goto branch; default: CHECK(COMPILE(ret, "branch condition", cond)); + branch: ADD_INSNL(ret, nd_line(cond), branchunless, else_label); ADD_INSNL(ret, nd_line(cond), jump, then_label); break; @@ -3705,6 +3711,31 @@ defined_expr(rb_iseq_t *iseq, LINK_ANCHO https://github.com/ruby/ruby/blob/trunk/compile.c#L3711 return done; } +static int +compile_defined_expr(rb_iseq_t *iseq, LINK_ANCHOR *const ret, NODE *node, VALUE needstr) +{ + const int line = nd_line(node); + if (!node->nd_head) { + VALUE str = rb_iseq_defined_string(DEFINED_NIL); + ADD_INSN1(ret, line, putobject, str); + } + else { + LABEL *lfinish[2]; + LINK_ELEMENT *last = ret->last; + lfinish[0] = NEW_LABEL(line); + lfinish[1] = 0; + defined_expr(iseq, ret, node->nd_head, lfinish, needstr); + if (lfinish[1]) { + INSERT_ELEM_NEXT(last, &new_insn_body(iseq, line, BIN(putnil), 0)->link); + ADD_INSN(ret, line, swap); + ADD_INSN(ret, line, pop); + ADD_LABEL(ret, lfinish[1]); + } + ADD_LABEL(ret, lfinish[0]); + } + return COMPILE_OK; +} + static VALUE make_name_for_block(const rb_iseq_t *orig_iseq) { @@ -6112,28 +6143,11 @@ iseq_compile_each(rb_iseq_t *iseq, LINK_ https://github.com/ruby/ruby/blob/trunk/compile.c#L6143 } break; } - case NODE_DEFINED:{ - if (popped) break; - if (!node->nd_head) { - VALUE str = rb_iseq_defined_string(DEFINED_NIL); - ADD_INSN1(ret, nd_line(node), putobject, str); - } - else { - LABEL *lfinish[2]; - LINK_ELEMENT *last = ret->last; - lfinish[0] = NEW_LABEL(line); - lfinish[1] = 0; - defined_expr(iseq, ret, node->nd_head, lfinish, Qtrue); - if (lfinish[1]) { - INSERT_ELEM_NEXT(last, &new_insn_body(iseq, line, BIN(putnil), 0)->link); - ADD_INSN(ret, line, swap); - ADD_INSN(ret, line, pop); - ADD_LABEL(ret, lfinish[1]); - } - ADD_LABEL(ret, lfinish[0]); + case NODE_DEFINED: + if (!popped) { + CHECK(compile_defined_expr(iseq, ret, node, Qtrue)); } break; - } case NODE_POSTEXE:{ /* compiled to: * ONCE{ rb_mRubyVMFrozenCore::core#set_postexe{ ... } } -- ML: ruby-changes@q... Info: http://www.atdot.net/~ko1/quickml/