From: Tony Cook Date: Sat, 17 Mar 2012 01:54:17 +0000 (+1100) Subject: fix another boundary case and hopefully improve performance X-Git-Tag: upstream/5.20.0~4576^2~3 X-Git-Url: http://review.tizen.org/git/?a=commitdiff_plain;h=a1aea1fe12c11cc8f3650979df95e88a810f3238;p=platform%2Fupstream%2Fperl.git fix another boundary case and hopefully improve performance The fix: the if we found ourselves at a charstart with only one character to read, readsize would be zero, handle that correctly. Performance: originally I read just the first byte of the next character, which meant as many extra read calls as there are characters left to read after the initial read. So O(Nleft) reads where Nleft is the number of characters left to read after the initial read. Now read as many bytes as there are characters left to read, which should mean the number of reads comes down to O(log(Nleft**2)) I think (but don't ask me to justify that.) --- diff --git a/sv.c b/sv.c index e2bed05..34dc534 100644 --- a/sv.c +++ b/sv.c @@ -7720,13 +7720,26 @@ S_sv_gets_read_record(pTHX_ SV *const sv, PerlIO *const fp, I32 append) } if (charcount < recsize) { - /* read the rest of the current character, and maybe the - beginning of the next, if we need it */ - STRLEN readsize = (charstart ? 0 : skip - (bend - bufp)) - + (charcount + 1 < recsize); + STRLEN readsize; STRLEN bufp_offset = bufp - buffer; SSize_t morebytesread; + /* originally I read enough to fill any incomplete + character and the first byte of the next + character if needed, but if there's many + multi-byte encoded characters we're going to be + making a read call for every character beyond + the original read size. + + So instead, read the rest of the character if + any, and enough bytes to match at least the + start bytes for each character we're going to + read. + */ + if (charstart) + readsize = recsize - charcount; + else + readsize = skip - (bend - bufp) + recsize - charcount - 1; buffer = SvGROW(sv, append + bytesread + readsize + 1) + append; bend = buffer + bytesread; morebytesread = PerlIO_read(fp, bend, readsize); diff --git a/t/io/utf8.t b/t/io/utf8.t index 919b734..29beba2 100644 --- a/t/io/utf8.t +++ b/t/io/utf8.t @@ -10,7 +10,7 @@ BEGIN { no utf8; # needed for use utf8 not griping about the raw octets -plan(tests => 59); +plan(tests => 61); $| = 1; @@ -354,6 +354,8 @@ is($failed, undef); open F, ">:utf8", $a_file; print F "foo\xE4"; print F "bar\xFE"; + print F "\xC0\xC8\xCC\xD2"; + print F "a\xE4ab"; print F "a\xE4a"; close F; open F, "<:utf8", $a_file; @@ -363,6 +365,10 @@ is($failed, undef); $line .= ; is($line, "foo\xE4bar\xFE", "rcatline with \$/ = \\4"); $line = ; + is($line, "\xC0\xC8\xCC\xD2", "readline with several encoded characters"); + $line = ; + is($line, "a\xE4ab", "readline with another boundary condition"); + $line = ; is($line, "a\xE4a", "readline with boundary condition"); close F;