diff options
author | akr <akr@b2dd03c8-39d4-4d8f-98ff-823fe69b080e> | 2008-01-03 17:48:06 +0000 |
---|---|---|
committer | akr <akr@b2dd03c8-39d4-4d8f-98ff-823fe69b080e> | 2008-01-03 17:48:06 +0000 |
commit | 52f9c1d2e173f07302ec263bf0026ded739a2d7f (patch) | |
tree | 8d310bc0d94fd3e8174031d6d30c1152d62790f5 /re.c | |
parent | 97751bbd5a3a6b66f91932ff3292f5b113ee964f (diff) | |
download | ruby-52f9c1d2e173f07302ec263bf0026ded739a2d7f.tar.gz |
* re.c (rb_reg_search): iterate onig_match for reverse mode.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@14876 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
Diffstat (limited to 're.c')
-rw-r--r-- | re.c | 31 |
1 files changed, 23 insertions, 8 deletions
@@ -1036,6 +1036,7 @@ rb_reg_search(VALUE re, VALUE str, int pos, int reverse) VALUE match; static struct re_registers regs; int range; + rb_encoding *enc = rb_enc_get(str); if (pos > RSTRING_LEN(str) || pos < 0) { rb_backref_set(Qnil); @@ -1045,19 +1046,33 @@ rb_reg_search(VALUE re, VALUE str, int pos, int reverse) rb_reg_prepare_re(re, str); if (reverse) { - range = -pos; + char *p = RSTRING_PTR(str) + pos; + while (1) { + result = onig_match(RREGEXP(re)->ptr, + (UChar*)(RSTRING_PTR(str)), + ((UChar*)(RSTRING_PTR(str)) + RSTRING_LEN(str)), + (UChar*)p, + ®s, + ONIG_OPTION_NONE); + if (result != ONIG_MISMATCH) { + result = p - RSTRING_PTR(str); + break; + } + if (RSTRING_PTR(str) == p) + break; + p = rb_enc_prev_char(RSTRING_PTR(str), p, enc); + } } else { range = RSTRING_LEN(str) - pos; + result = onig_search(RREGEXP(re)->ptr, + (UChar*)(RSTRING_PTR(str)), + ((UChar*)(RSTRING_PTR(str)) + RSTRING_LEN(str)), + ((UChar*)(RSTRING_PTR(str)) + pos), + ((UChar*)(RSTRING_PTR(str)) + pos + range), + ®s, ONIG_OPTION_NONE); } - result = onig_search(RREGEXP(re)->ptr, - (UChar*)(RSTRING_PTR(str)), - ((UChar*)(RSTRING_PTR(str)) + RSTRING_LEN(str)), - ((UChar*)(RSTRING_PTR(str)) + pos), - ((UChar*)(RSTRING_PTR(str)) + pos + range), - ®s, ONIG_OPTION_NONE); - if (result < 0) { if (result == ONIG_MISMATCH) { rb_backref_set(Qnil); |