Piotr Caban : msvcrt: Handle read buffer flushing in msvcrt_flush_buffer helper.

Alexandre Julliard julliard at wine.codeweavers.com
Mon Jan 26 15:59:19 CST 2015


Module: wine
Branch: master
Commit: 7880525f442f548d2615797fab37e0e342e0b302
URL:    http://source.winehq.org/git/wine.git/?a=commit;h=7880525f442f548d2615797fab37e0e342e0b302

Author: Piotr Caban <piotr at codeweavers.com>
Date:   Fri Jan 23 17:27:23 2015 +0100

msvcrt: Handle read buffer flushing in msvcrt_flush_buffer helper.

---

 dlls/msvcrt/file.c       | 49 +++++++++++++++++-------------------------------
 dlls/msvcrt/tests/file.c | 14 ++++++++++++++
 2 files changed, 31 insertions(+), 32 deletions(-)

diff --git a/dlls/msvcrt/file.c b/dlls/msvcrt/file.c
index 8c59239..9ecf129 100644
--- a/dlls/msvcrt/file.c
+++ b/dlls/msvcrt/file.c
@@ -606,16 +606,18 @@ void msvcrt_init_io(void)
 /* INTERNAL: Flush stdio file buffer */
 static int msvcrt_flush_buffer(MSVCRT_FILE* file)
 {
-  if(file->_flag & (MSVCRT__IOMYBUF | MSVCRT__USERBUF)) {
+    if((file->_flag & (MSVCRT__IOREAD|MSVCRT__IOWRT)) == MSVCRT__IOWRT &&
+            file->_flag & (MSVCRT__IOMYBUF|MSVCRT__USERBUF)) {
         int cnt=file->_ptr-file->_base;
         if(cnt>0 && MSVCRT__write(file->_file, file->_base, cnt) != cnt) {
             file->_flag |= MSVCRT__IOERR;
             return MSVCRT_EOF;
         }
-        file->_ptr=file->_base;
-        file->_cnt=0;
-  }
-  return 0;
+    }
+
+    file->_ptr=file->_base;
+    file->_cnt=0;
+    return 0;
 }
 
 /*********************************************************************
@@ -964,23 +966,17 @@ int CDECL MSVCRT_fflush(MSVCRT_FILE* file)
  */
 int CDECL MSVCRT__fflush_nolock(MSVCRT_FILE* file)
 {
+    int res;
+
     if(!file) {
         msvcrt_flush_all_buffers(MSVCRT__IOWRT);
-    } else if(file->_flag & MSVCRT__IOWRT) {
-        int res;
-
-        res = msvcrt_flush_buffer(file);
-        if(!res && (file->_flag & MSVCRT__IOCOMMIT))
-            res = MSVCRT__commit(file->_file) ? MSVCRT_EOF : 0;
-
-        return res;
-    } else if(file->_flag & MSVCRT__IOREAD) {
-        file->_cnt = 0;
-        file->_ptr = file->_base;
-
         return 0;
     }
-    return 0;
+
+    res = msvcrt_flush_buffer(file);
+    if(!res && (file->_flag & MSVCRT__IOCOMMIT))
+        res = MSVCRT__commit(file->_file) ? MSVCRT_EOF : 0;
+    return res;
 }
 
 /*********************************************************************
@@ -1333,18 +1329,13 @@ int CDECL MSVCRT__fseeki64_nolock(MSVCRT_FILE* file, __int64 offset, int whence)
 {
   int ret;
 
-  /* Flush output if needed */
-  if(file->_flag & MSVCRT__IOWRT)
-	msvcrt_flush_buffer(file);
-
   if(whence == SEEK_CUR && file->_flag & MSVCRT__IOREAD ) {
       whence = SEEK_SET;
       offset += MSVCRT__ftelli64_nolock(file);
   }
 
-  /* Discard buffered input */
-  file->_cnt = 0;
-  file->_ptr = file->_base;
+  /* Flush output if needed */
+  msvcrt_flush_buffer(file);
   /* Reset direction of i/o */
   if(file->_flag & MSVCRT__IORW) {
         file->_flag &= ~(MSVCRT__IOREAD|MSVCRT__IOWRT);
@@ -4369,14 +4360,8 @@ int CDECL MSVCRT_fsetpos(MSVCRT_FILE* file, MSVCRT_fpos_t *pos)
   int ret;
 
   MSVCRT__lock_file(file);
-  /* Note that all this has been lifted 'as is' from fseek */
-  if(file->_flag & MSVCRT__IOWRT)
-	msvcrt_flush_buffer(file);
+  msvcrt_flush_buffer(file);
 
-  /* Discard buffered input */
-  file->_cnt = 0;
-  file->_ptr = file->_base;
-  
   /* Reset direction of i/o */
   if(file->_flag & MSVCRT__IORW) {
         file->_flag &= ~(MSVCRT__IOREAD|MSVCRT__IOWRT);
diff --git a/dlls/msvcrt/tests/file.c b/dlls/msvcrt/tests/file.c
index c253f88..860db6a 100644
--- a/dlls/msvcrt/tests/file.c
+++ b/dlls/msvcrt/tests/file.c
@@ -2244,6 +2244,7 @@ static void test_write_flush_size(FILE *file, int bufsize)
     char *inbuffer;
     char *outbuffer;
     int size, fd;
+    fpos_t pos, pos2;
 
     fd = fileno(file);
     inbuffer = calloc(bufsize + 1, 1);
@@ -2275,6 +2276,19 @@ static void test_write_flush_size(FILE *file, int bufsize)
     fseek(file, 0, SEEK_SET);
     ok(fread(inbuffer, 1, bufsize, file) == bufsize, "read failed\n");
     ok(memcmp(outbuffer, inbuffer, bufsize) != 0, "unexpected flush by %d/2 byte double write\n", bufsize);
+
+    ok(!fseek(file, -1, SEEK_END), "fseek failed\n");
+    ok(!fgetpos(file, &pos), "fgetpos failed\n");
+    ok(fread(inbuffer, 1, 1, file) == 1, "fread failed\n");
+    ok(file->_flag & _IOREAD, "file->_flag = %x\n", file->_flag);
+    ok(!file->_cnt, "file->_cnt = %d\n", file->_cnt);
+    ok(file->_ptr != file->_base, "file->_ptr == file->_base\n");
+    ok(fwrite(outbuffer, 1, bufsize, file), "fwrite failed\n");
+    ok(file->_flag & _IOREAD, "file->_flag = %x\n", file->_flag);
+    ok(!file->_cnt, "file->_cnt = %d\n", file->_cnt);
+    ok(file->_ptr == file->_base, "file->_ptr == file->_base\n");
+    ok(!fgetpos(file, &pos2), "fgetpos failed\n");
+    ok(pos+bufsize+1 == pos2, "pos = %d (%d)\n", (int)pos, (int)pos2);
     free(inbuffer);
     free(outbuffer);
 }




More information about the wine-cvs mailing list