ID: 22362 Updated by: [EMAIL PROTECTED] Reported By: php at codewhore dot org Status: Assigned Bug Type: Filesystem function related Operating System: Linux 2.4 PHP Version: 4.3.1 Assigned To: wez New Comment:
Save yourself some energy :) I'm going to look into this this weekend; I know precisely how to fix this, and it most likely not something you would have time to think of by the time I've sorted it out :) I will update this report when it is fixed and you will be able to try the fix using a stable snapshot. Thanks for your efforts so far! Previous Comments: ------------------------------------------------------------------------ [2003-02-21 17:44:27] php at codewhore dot org FWIW, the problem also occurs in the following piece of code: <?php $fp = fopen('./foo', 'w+'); fwrite($fp, 'quxbar', 6); fseek($fp, 3, SEEK_SET); fread($fp, 1); fwrite($fp, '!', 1); fseek($fp, 0, SEEK_SET); $buf = fread($fp, 4095); echo "$buf\n"; ?> There's no fseek() between the fread() and fwrite(), so my previous theory seems to be quite wrong. I'll do more digging as time permits... ------------------------------------------------------------------------ [2003-02-21 17:02:51] php at codewhore dot org I'm still mostly lost, but commenting out the following segment of code fixes the problem for the test case. Obviously this isn't the solution, but I think I have it narrowed down to _php_stream_seek thinking that the underlying file position matches the seek offset when it doesn't in reality. PHPAPI int _php_stream_seek(php_stream *stream, off_t offset, int whence TSRMLS_DC) { /* not moving anywhere */ #if 0 if ((offset == 0 && whence == SEEK_CUR) || (offset == stream->position && whence == SEEK_SET)) return 0; #endif ------------------------------------------------------------------------ [2003-02-21 14:43:43] php at codewhore dot org I've verified that this behavior still exists in today's snapshot (php4-STABLE-200302212030.tar.gz). ------------------------------------------------------------------------ [2003-02-21 13:58:03] php at codewhore dot org In PHP: <?php $fp = fopen('./foo', 'w+'); fwrite($fp, 'quxbar', 6); fseek($fp, 3, SEEK_SET); fread($fp, 1); fseek($fp, 4, SEEK_SET); fwrite($fp, '!', 1); fseek($fp, 0, SEEK_SET); $buf = fread($fp, 4095); echo "$buf\n"; ?> In C: #include <stdio.h> int main() { static char buf[4096]; FILE *f = fopen("./foo", "w+"); fwrite("quxbar", 6, 1, f); fseek(f, 3, SEEK_SET); fread(&buf, 1, 1, f); fseek(f, 4, SEEK_SET); fwrite("!", 1, 1, f); fseek(f, 0, SEEK_SET); fread(&buf, 4095, 1, f); printf("%s\n", buf); return 0; } The PHP version outputs 'quxbar!. The C version outputs 'quxb!r'. In fact, for any PHP code of the structure: <?php fseek($fp, $x, SEEK_SET); fread($fp, $n); fseek($fp, $y, SEEK_SET); fwrite($fp, '!', 1); ?> the data from the fwrite always gets written to the end of the stream when $x < $y and $n is exactly 1. I can reproduce this with different whence parameters, but it seems to only happen with a seek followed by a single-byte read. ------------------------------------------------------------------------ -- Edit this bug report at http://bugs.php.net/?id=22362&edit=1