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

Reply via email to