/* * Unit test suite for file functions * * Copyright 2002 Bill Currie * Copyright 2005 Paul Rupe * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include "wine/test.h" #include #include #include #include #include #include #include #include #include #include #include #include #include "msvcrt.h" static void test_fdopen( void ) { static const char buffer[] = {0,1,2,3,4,5,6,7,8,9}; char ibuf[10]; int fd; FILE *file; fd = open ("fdopen.tst", O_WRONLY | O_CREAT | O_BINARY, _S_IREAD |_S_IWRITE); write (fd, buffer, sizeof (buffer)); close (fd); fd = open ("fdopen.tst", O_RDONLY | O_BINARY); lseek (fd, 5, SEEK_SET); file = fdopen (fd, "rb"); ok (fread (ibuf, 1, sizeof (buffer), file) == 5, "read wrong byte count\n"); ok (memcmp (ibuf, buffer + 5, 5) == 0, "read wrong bytes\n"); fclose (file); unlink ("fdopen.tst"); } static void test_fileops( void ) { static const char outbuffer[] = "0,1,2,3,4,5,6,7,8,9"; char buffer[256]; WCHAR wbuffer[256]; int fd; FILE *file; fpos_t pos; int i, c; fd = open ("fdopen.tst", O_WRONLY | O_CREAT | O_BINARY, _S_IREAD |_S_IWRITE); write (fd, outbuffer, sizeof (outbuffer)); close (fd); fd = open ("fdopen.tst", O_RDONLY | O_BINARY); file = fdopen (fd, "rb"); ok(strlen(outbuffer) == (sizeof(outbuffer)-1),"strlen/sizeof error\n"); ok(fgets(buffer,sizeof(buffer),file) !=0,"fgets failed unexpected\n"); ok(fgets(buffer,sizeof(buffer),file) ==0,"fgets didn't signal EOF\n"); ok(feof(file) !=0,"feof doesn't signal EOF\n"); rewind(file); ok(fgets(buffer,strlen(outbuffer),file) !=0,"fgets failed unexpected\n"); ok(lstrlenA(buffer) == lstrlenA(outbuffer) -1,"fgets didn't read right size\n"); ok(fgets(buffer,sizeof(outbuffer),file) !=0,"fgets failed unexpected\n"); ok(strlen(buffer) == 1,"fgets dropped chars\n"); ok(buffer[0] == outbuffer[strlen(outbuffer)-1],"fgets exchanged chars\n"); rewind(file); for (i = 0, c = EOF; i < sizeof(outbuffer); i++) { ok((c = fgetc(file)) == outbuffer[i], "fgetc returned wrong data\n"); } ok((c = fgetc(file)) == EOF, "getc did not return EOF\n"); ok(feof(file), "feof did not return EOF\n"); ok(ungetc(c, file) == EOF, "ungetc(EOF) did not return EOF\n"); ok(feof(file), "feof after ungetc(EOF) did not return EOF\n"); ok((c = fgetc(file)) == EOF, "getc did not return EOF\n"); c = outbuffer[sizeof(outbuffer) - 1]; ok(ungetc(c, file) == c, "ungetc did not return its input\n"); ok(!feof(file), "feof after ungetc returned EOF\n"); ok((c = fgetc(file)) != EOF, "getc after ungetc returned EOF\n"); ok(c == outbuffer[sizeof(outbuffer) - 1], "getc did not return ungetc'd data\n"); ok(!feof(file), "feof after getc returned EOF prematurely\n"); ok((c = fgetc(file)) == EOF, "getc did not return EOF\n"); ok(feof(file), "feof after getc did not return EOF\n"); rewind(file); ok(fgetpos(file,&pos) == 0, "fgetpos failed unexpected\n"); ok(pos == 0, "Unexpected result of fgetpos 0x%Lx\n", pos); pos = (ULONGLONG)sizeof (outbuffer); ok(fsetpos(file, &pos) == 0, "fsetpos failed unexpected\n"); ok(fgetpos(file,&pos) == 0, "fgetpos failed unexpected\n"); ok(pos == (ULONGLONG)sizeof (outbuffer), "Unexpected result of fgetpos 0x%Lx\n", pos); fclose (file); fd = open ("fdopen.tst", O_RDONLY | O_TEXT); file = fdopen (fd, "rt"); /* open in TEXT mode */ ok(fgetws(wbuffer,sizeof(wbuffer),file) !=0,"fgetws failed unexpected\n"); ok(fgetws(wbuffer,sizeof(wbuffer),file) ==0,"fgetws didn't signal EOF\n"); ok(feof(file) !=0,"feof doesn't signal EOF\n"); rewind(file); ok(fgetws(wbuffer,strlen(outbuffer),file) !=0,"fgetws failed unexpected\n"); ok(lstrlenW(wbuffer) == (lstrlenA(outbuffer) -1),"fgetws didn't read right size\n"); ok(fgetws(wbuffer,sizeof(outbuffer),file) !=0,"fgets failed unexpected\n"); ok(lstrlenW(wbuffer) == 1,"fgets dropped chars\n"); fclose (file); file = fopen("fdopen.tst", "rb"); ok( file != NULL, "fopen failed\n"); /* sizeof(buffer) > content of file */ ok(fread(buffer, sizeof(buffer), 1, file) == 0, "fread test failed\n"); /* feof should be set now */ ok(feof(file), "feof after fread failed\n"); fclose (file); unlink ("fdopen.tst"); } #define IOMODE (ao?"ascii mode":"binary mode") static void test_readmode( BOOL ascii_mode ) { static const char outbuffer[] = "0,1,2,3,4,5,6,7,8,9\r\n\r\nA,B,C,D,E\r\nX,Y,Z"; static const char padbuffer[] = "ghjghjghjghj"; static const char nlbuffer[] = "\r\n"; char buffer[2*MSVCRT_BUFSIZ+256], *optr; int fd; FILE *file; int i, j, m, fp, ao, *ip, pl; long l; fd = open ("fdopen.tst", O_WRONLY | O_CREAT | O_BINARY, _S_IREAD |_S_IWRITE); /* an internal buffer of MSVCRT_BUFSIZ is maintained, so make a file big * enough to test operations that cross the buffer boundary */ j = (2*MSVCRT_BUFSIZ-4)/strlen(padbuffer); for (i=0; i 0, "Couldn't open test file\n" ); count = _write( fd, temptext, sizeof(temptext) ); ok( count > 0, "Couldn't write to test file\n" ); /* get current file pointer */ cur = _lseek( fd, 0, SEEK_CUR ); /* make the file smaller */ ok( _chsize( fd, sizeof(temptext) / 2 ) == 0, "_chsize() failed\n" ); pos = _lseek( fd, 0, SEEK_CUR ); ok( cur == pos, "File pointer changed from: %ld to: %ld\n", cur, pos ); ok( _filelength( fd ) == sizeof(temptext) / 2, "Wrong file size\n" ); /* enlarge the file */ ok( _chsize( fd, sizeof(temptext) * 2 ) == 0, "_chsize() failed\n" ); pos = _lseek( fd, 0, SEEK_CUR ); ok( cur == pos, "File pointer changed from: %ld to: %ld\n", cur, pos ); ok( _filelength( fd ) == sizeof(temptext) * 2, "Wrong file size\n" ); _close( fd ); _unlink( tempfile ); } static void test_fopen_fclose_fcloseall( void ) { char fname1[] = "empty1"; char fname2[] = "empty2"; char fname3[] = "empty3"; FILE *stream1, *stream2, *stream3, *stream4; int ret, numclosed; /* testing fopen() */ stream1 = fopen(fname1, "w+"); ok(stream1 != NULL, "The file '%s' was not opened\n", fname1); stream2 = fopen(fname2, "w "); ok(stream2 != NULL, "The file '%s' was not opened\n", fname2 ); _unlink(fname3); stream3 = fopen(fname3, "r"); ok(stream3 == NULL, "The file '%s' shouldn't exist before\n", fname3 ); stream3 = fopen(fname3, "w+"); ok(stream3 != NULL, "The file '%s' should be opened now\n", fname3 ); errno = 0xfaceabad; stream4 = fopen("", "w+"); ok(stream4 == NULL && errno == ENOENT, "filename is empty, errno = %d (expected 2)\n", errno); errno = 0xfaceabad; stream4 = fopen(NULL, "w+"); ok(stream4 == NULL && (errno == EINVAL || errno == ENOENT), "filename is NULL, errno = %d (expected 2 or 22)\n", errno); /* testing fclose() */ ret = fclose(stream2); ok(ret == 0, "The file '%s' was not closed\n", fname2); ret = fclose(stream3); ok(ret == 0, "The file '%s' was not closed\n", fname3); ret = fclose(stream2); ok(ret == EOF, "Closing file '%s' returned %d\n", fname2, ret); ret = fclose(stream3); ok(ret == EOF, "Closing file '%s' returned %d\n", fname3, ret); /* testing fcloseall() */ numclosed = _fcloseall(); /* fname1 should be closed here */ ok(numclosed == 1, "Number of files closed by fcloseall(): %u\n", numclosed); numclosed = _fcloseall(); ok(numclosed == 0, "Number of files closed by fcloseall(): %u\n", numclosed); ok(_unlink(fname1) == 0, "Couldn't unlink file named '%s'\n", fname1); ok(_unlink(fname2) == 0, "Couldn't unlink file named '%s'\n", fname2); ok(_unlink(fname3) == 0, "Couldn't unlink file named '%s'\n", fname3); } static void test_get_osfhandle(void) { int fd; char fname[] = "t_get_osfhanle"; DWORD bytes_written; HANDLE handle; fd = _sopen(fname, _O_CREAT|_O_RDWR, _SH_DENYRW, _S_IREAD | _S_IWRITE); handle = (HANDLE)_get_osfhandle(fd); WriteFile(handle, "bar", 3, &bytes_written, NULL); _close(fd); fd = _open(fname, _O_RDONLY, 0); ok(fd != -1, "Coudn't open '%s' after _get_osfhanle()\n", fname); _close(fd); _unlink(fname); } static void test_setmaxstdio(void) { ok(2048 == _setmaxstdio(2048),"_setmaxstdio returned %d instead of 2048\n",_setmaxstdio(2048)); ok(-1 == _setmaxstdio(2049),"_setmaxstdio returned %d instead of -1\n",_setmaxstdio(2049)); } START_TEST(file) { int arg_c; char** arg_v; arg_c = winetest_get_mainargs( &arg_v ); /* testing low-level I/O */ if (arg_c >= 3) { if (arg_c == 3) test_file_inherit_child(arg_v[2]); else test_file_inherit_child_no(arg_v[2]); return; } test_file_inherit(arg_v[0]); test_file_write_read(); test_chsize(); /* testing stream I/O */ test_fdopen(); test_fopen_fclose_fcloseall(); test_fileops(); test_readmode(FALSE); /* binary mode */ test_readmode(TRUE); /* ascii mode */ test_fgetc(); test_fgetwc(); test_ctrlz(); test_file_put_get(); test_tmpnam(); test_get_osfhandle(); test_setmaxstdio(); }