xref: /reactos/sdk/lib/crt/stdio/file.c (revision 23373acb)
1 /*
2  * PROJECT:         ReactOS CRT library
3  * LICENSE:         LGPL - See COPYING in the top level directory
4  * FILE:            lib/sdk/crt/stdio/file.c
5  * PURPOSE:         File CRT functions
6  * PROGRAMMERS:     Wine team
7  *                  Ported to ReactOS by Aleksey Bragin (aleksey@reactos.org)
8  */
9 
10 /*********************************************
11  * This file contains ReactOS changes!!
12  * Don't blindly sync it with Wine code!
13  *
14  * If you break Unicode output on the console again, please update this counter:
15  *   int hours_wasted_on_this = 42;
16  *********************************************/
17 
18 /*
19  * msvcrt.dll file functions
20  *
21  * Copyright 1996,1998 Marcus Meissner
22  * Copyright 1996 Jukka Iivonen
23  * Copyright 1997,2000 Uwe Bonnes
24  * Copyright 2000 Jon Griffiths
25  * Copyright 2004 Eric Pouech
26  * Copyright 2004 Juan Lang
27  *
28  * This library is free software; you can redistribute it and/or
29  * modify it under the terms of the GNU Lesser General Public
30  * License as published by the Free Software Foundation; either
31  * version 2.1 of the License, or (at your option) any later version.
32  *
33  * This library is distributed in the hope that it will be useful,
34  * but WITHOUT ANY WARRANTY; without even the implied warranty of
35  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
36  * Lesser General Public License for more details.
37  *
38  * You should have received a copy of the GNU Lesser General Public
39  * License along with this library; if not, write to the Free Software
40  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
41  *
42  * TODO
43  * Use the file flag hints O_SEQUENTIAL, O_RANDOM, O_SHORT_LIVED
44  */
45 
46 #include <precomp.h>
47 #include "wine/unicode.h"
48 #include "internal/wine/msvcrt.h"
49 
50 #include <sys/utime.h>
51 #include <direct.h>
52 
53 int *__p__fmode(void);
54 int *__p___mb_cur_max(void);
55 
56 extern int _commode;
57 
58 #ifndef _IOCOMMIT
59 #define _IOCOMMIT 0x4000
60 #endif
61 
62 #ifdef feof
63 #undef feof
64 #endif
65 #ifdef _fileno
66 #undef _fileno
67 #endif
68 #ifdef ferror
69 #undef ferror
70 #endif
71 #ifdef clearerr
72 #undef clearerr
73 #endif
74 
75 #undef getc
76 #undef getwc
77 #undef getchar
78 #undef getwchar
79 #undef putc
80 #undef putwc
81 #undef putchar
82 #undef putwchar
83 
84 #undef vprintf
85 #undef vwprintf
86 
87 /* _access() bit flags FIXME: incomplete */
88 /* defined in crt/io.h */
89 
90 /* values for wxflag in file descriptor */
91 #define WX_OPEN           0x01
92 #define WX_ATEOF          0x02
93 #define WX_READNL         0x04  /* read started with \n */
94 #define WX_READEOF        0x04  /* like ATEOF, but for underlying file rather than buffer */
95 #define WX_PIPE           0x08
96 #define WX_READCR         0x08  /* underlying file is at \r */
97 #define WX_DONTINHERIT    0x10
98 #define WX_APPEND         0x20
99 #define WX_TTY            0x40
100 #define WX_TEXT           0x80
101 
102 /* values for exflag - it's used differently in msvcr90.dll*/
103 #define EF_UTF8           0x01
104 #define EF_UTF16          0x02
105 #define EF_CRIT_INIT      0x04
106 #define EF_UNK_UNICODE    0x08
107 
108 static char utf8_bom[3] = { 0xef, 0xbb, 0xbf };
109 static char utf16_bom[2] = { 0xff, 0xfe };
110 
111 /* FIXME: this should be allocated dynamically */
112 #define MSVCRT_MAX_FILES 2048
113 #define MSVCRT_FD_BLOCK_SIZE 32
114 
115 #define MSVCRT_INTERNAL_BUFSIZ 4096
116 
117 /*********************************************************************
118  *		__pioinfo (MSVCRT.@)
119  * array of pointers to ioinfo arrays [32]
120  */
121 ioinfo * __pioinfo[MSVCRT_MAX_FILES/MSVCRT_FD_BLOCK_SIZE] = { 0 };
122 
123 /*********************************************************************
124  *		__badioinfo (MSVCRT.@)
125  */
126 ioinfo __badioinfo = { INVALID_HANDLE_VALUE, WX_TEXT };
127 
128 typedef struct {
129     FILE file;
130     CRITICAL_SECTION crit;
131 } file_crit;
132 
133 FILE _iob[_IOB_ENTRIES] = { { 0 } };
134 static file_crit* MSVCRT_fstream[MSVCRT_MAX_FILES/MSVCRT_FD_BLOCK_SIZE] = { NULL };
135 static int MSVCRT_max_streams = 512, MSVCRT_stream_idx;
136 
137 /* INTERNAL: process umask */
138 static int MSVCRT_umask = 0;
139 
140 /* INTERNAL: static data for tmpnam and _wtmpname functions */
141 static int tmpnam_unique;
142 
143 /* This critical section protects the MSVCRT_fstreams table
144  * and MSVCRT_stream_idx from race conditions. It also
145  * protects fd critical sections creation code.
146  */
147 static CRITICAL_SECTION MSVCRT_file_cs;
148 static CRITICAL_SECTION_DEBUG MSVCRT_file_cs_debug =
149 {
150     0, 0, &MSVCRT_file_cs,
151     { &MSVCRT_file_cs_debug.ProcessLocksList, &MSVCRT_file_cs_debug.ProcessLocksList },
152       0, 0, { (DWORD_PTR)(__FILE__ ": MSVCRT_file_cs") }
153 };
154 static CRITICAL_SECTION MSVCRT_file_cs = { &MSVCRT_file_cs_debug, -1, 0, 0, 0, 0 };
155 #define LOCK_FILES()    do { EnterCriticalSection(&MSVCRT_file_cs); } while (0)
156 #define UNLOCK_FILES()  do { LeaveCriticalSection(&MSVCRT_file_cs); } while (0)
157 
158 static inline ioinfo* get_ioinfo_nolock(int fd)
159 {
160     ioinfo *ret = NULL;
161     if(fd>=0 && fd<MSVCRT_MAX_FILES)
162         ret = __pioinfo[fd/MSVCRT_FD_BLOCK_SIZE];
163     if(!ret)
164         return &__badioinfo;
165 
166     return ret + (fd%MSVCRT_FD_BLOCK_SIZE);
167 }
168 
169 static inline void init_ioinfo_cs(ioinfo *info)
170 {
171     if(!(info->exflag & EF_CRIT_INIT)) {
172         LOCK_FILES();
173         if(!(info->exflag & EF_CRIT_INIT)) {
174             InitializeCriticalSection(&info->crit);
175             info->exflag |= EF_CRIT_INIT;
176         }
177         UNLOCK_FILES();
178     }
179 }
180 
181 ioinfo* get_ioinfo(int fd)
182 {
183     ioinfo *ret = get_ioinfo_nolock(fd);
184     if(ret == &__badioinfo)
185         return ret;
186     init_ioinfo_cs(ret);
187     EnterCriticalSection(&ret->crit);
188     return ret;
189 }
190 
191 static inline BOOL alloc_pioinfo_block(int fd)
192 {
193     ioinfo *block;
194     int i;
195 
196     if(fd<0 || fd>=MSVCRT_MAX_FILES)
197     {
198         *_errno() = ENFILE;
199         return FALSE;
200     }
201 
202     block = calloc(MSVCRT_FD_BLOCK_SIZE, sizeof(ioinfo));
203     if(!block)
204     {
205         WARN(":out of memory!\n");
206         *_errno() = ENOMEM;
207         return FALSE;
208     }
209     for(i=0; i<MSVCRT_FD_BLOCK_SIZE; i++)
210         block[i].handle = INVALID_HANDLE_VALUE;
211     if(InterlockedCompareExchangePointer((void**)&__pioinfo[fd/MSVCRT_FD_BLOCK_SIZE], block, NULL))
212         free(block);
213     return TRUE;
214 }
215 
216 static inline ioinfo* get_ioinfo_alloc_fd(int fd)
217 {
218     ioinfo *ret;
219 
220     ret = get_ioinfo(fd);
221     if(ret != &__badioinfo)
222         return ret;
223 
224     if(!alloc_pioinfo_block(fd))
225         return &__badioinfo;
226 
227     return get_ioinfo(fd);
228 }
229 
230 static inline ioinfo* get_ioinfo_alloc(int *fd)
231 {
232     int i;
233 
234     *fd = -1;
235     for(i=0; i<MSVCRT_MAX_FILES; i++)
236     {
237         ioinfo *info = get_ioinfo_nolock(i);
238 
239         if(info == &__badioinfo)
240         {
241             if(!alloc_pioinfo_block(i))
242                 return &__badioinfo;
243             info = get_ioinfo_nolock(i);
244         }
245 
246         init_ioinfo_cs(info);
247         if(TryEnterCriticalSection(&info->crit))
248         {
249             if(info->handle == INVALID_HANDLE_VALUE)
250             {
251                 *fd = i;
252                 return info;
253             }
254             LeaveCriticalSection(&info->crit);
255         }
256     }
257 
258     WARN(":files exhausted!\n");
259     *_errno() = ENFILE;
260     return &__badioinfo;
261 }
262 
263 void release_ioinfo(ioinfo *info)
264 {
265     if(info!=&__badioinfo && info->exflag & EF_CRIT_INIT)
266         LeaveCriticalSection(&info->crit);
267 }
268 
269 static inline FILE* msvcrt_get_file(int i)
270 {
271     file_crit *ret;
272 
273     if(i >= MSVCRT_max_streams)
274         return NULL;
275 
276     if(i < _IOB_ENTRIES)
277         return &_iob[i];
278 
279     ret = MSVCRT_fstream[i/MSVCRT_FD_BLOCK_SIZE];
280     if(!ret) {
281         MSVCRT_fstream[i/MSVCRT_FD_BLOCK_SIZE] = calloc(MSVCRT_FD_BLOCK_SIZE, sizeof(file_crit));
282         if(!MSVCRT_fstream[i/MSVCRT_FD_BLOCK_SIZE]) {
283             ERR("out of memory\n");
284             *_errno() = ENOMEM;
285             return NULL;
286         }
287 
288         ret = MSVCRT_fstream[i/MSVCRT_FD_BLOCK_SIZE] + (i%MSVCRT_FD_BLOCK_SIZE);
289     } else
290         ret += i%MSVCRT_FD_BLOCK_SIZE;
291 
292     return &ret->file;
293 }
294 
295 /* INTERNAL: free a file entry fd */
296 static void msvcrt_free_fd(int fd)
297 {
298   ioinfo *fdinfo = get_ioinfo(fd);
299 
300   if(fdinfo != &__badioinfo)
301   {
302     fdinfo->handle = INVALID_HANDLE_VALUE;
303     fdinfo->wxflag = 0;
304   }
305   TRACE(":fd (%d) freed\n",fd);
306 
307   if (fd < 3)
308   {
309     switch (fd)
310     {
311     case 0:
312         SetStdHandle(STD_INPUT_HANDLE, 0);
313         break;
314     case 1:
315         SetStdHandle(STD_OUTPUT_HANDLE, 0);
316         break;
317     case 2:
318         SetStdHandle(STD_ERROR_HANDLE, 0);
319         break;
320     }
321   }
322   release_ioinfo(fdinfo);
323 }
324 
325 static void msvcrt_set_fd(ioinfo *fdinfo, HANDLE hand, int flag)
326 {
327   fdinfo->handle = hand;
328   fdinfo->wxflag = WX_OPEN | (flag & (WX_DONTINHERIT | WX_APPEND | WX_TEXT | WX_PIPE | WX_TTY));
329   fdinfo->lookahead[0] = '\n';
330   fdinfo->lookahead[1] = '\n';
331   fdinfo->lookahead[2] = '\n';
332   fdinfo->exflag &= EF_CRIT_INIT;
333 
334   switch (fdinfo-__pioinfo[0])
335   {
336   case 0: SetStdHandle(STD_INPUT_HANDLE,  hand); break;
337   case 1: SetStdHandle(STD_OUTPUT_HANDLE, hand); break;
338   case 2: SetStdHandle(STD_ERROR_HANDLE,  hand); break;
339   }
340 }
341 
342 /* INTERNAL: Allocate an fd slot from a Win32 HANDLE */
343 /*static*/ int msvcrt_alloc_fd(HANDLE hand, int flag)
344 {
345     int fd;
346     ioinfo *info = get_ioinfo_alloc(&fd);
347 
348     TRACE(":handle (%p) allocating fd (%d)\n", hand, fd);
349 
350     if(info == &__badioinfo)
351         return -1;
352 
353     msvcrt_set_fd(info, hand, flag);
354     release_ioinfo(info);
355     return fd;
356 }
357 
358 /* INTERNAL: Allocate a FILE* for an fd slot */
359 /* caller must hold the files lock */
360 static FILE* msvcrt_alloc_fp(void)
361 {
362   int i;
363   FILE *file;
364 
365   for (i = 3; i < MSVCRT_max_streams; i++)
366   {
367     file = msvcrt_get_file(i);
368     if (!file)
369       return NULL;
370 
371     if (file->_flag == 0)
372     {
373       if (i == MSVCRT_stream_idx) MSVCRT_stream_idx++;
374       return file;
375     }
376   }
377 
378   return NULL;
379 }
380 
381 /* INTERNAL: initialize a FILE* from an open fd */
382 static int msvcrt_init_fp(FILE* file, int fd, unsigned stream_flags)
383 {
384   TRACE(":fd (%d) allocating FILE*\n",fd);
385   if (!(get_ioinfo_nolock(fd)->wxflag & WX_OPEN))
386   {
387     WARN(":invalid fd %d\n",fd);
388     *__doserrno() = 0;
389     *_errno() = EBADF;
390     return -1;
391   }
392   file->_ptr = file->_base = NULL;
393   file->_cnt = 0;
394   file->_file = fd;
395   file->_flag = stream_flags;
396   file->_tmpfname = NULL;
397 
398   if(file<_iob || file>=_iob+_IOB_ENTRIES)
399       InitializeCriticalSection(&((file_crit*)file)->crit);
400 
401   TRACE(":got FILE* (%p)\n",file);
402   return 0;
403 }
404 
405 /* INTERNAL: Create an inheritance data block (for spawned process)
406  * The inheritance block is made of:
407  *      00      int     nb of file descriptor (NBFD)
408  *      04      char    file flags (wxflag): repeated for each fd
409  *      4+NBFD  HANDLE  file handle: repeated for each fd
410  */
411 unsigned create_io_inherit_block(WORD *size, BYTE **block)
412 {
413   int         fd, last_fd;
414   char*       wxflag_ptr;
415   HANDLE*     handle_ptr;
416   ioinfo*     fdinfo;
417 
418   for (last_fd=MSVCRT_MAX_FILES-1; last_fd>=0; last_fd--)
419     if (get_ioinfo_nolock(last_fd)->handle != INVALID_HANDLE_VALUE)
420       break;
421   last_fd++;
422 
423   *size = sizeof(unsigned) + (sizeof(char) + sizeof(HANDLE)) * last_fd;
424   *block = calloc(1, *size);
425   if (!*block)
426   {
427     *size = 0;
428     return FALSE;
429   }
430   wxflag_ptr = (char*)*block + sizeof(unsigned);
431   handle_ptr = (HANDLE*)(wxflag_ptr + last_fd);
432 
433   *(unsigned*)*block = last_fd;
434   for (fd = 0; fd < last_fd; fd++)
435   {
436     /* to be inherited, we need it to be open, and that DONTINHERIT isn't set */
437     fdinfo = get_ioinfo(fd);
438     if ((fdinfo->wxflag & (WX_OPEN | WX_DONTINHERIT)) == WX_OPEN)
439     {
440       *wxflag_ptr = fdinfo->wxflag;
441       *handle_ptr = fdinfo->handle;
442     }
443     else
444     {
445       *wxflag_ptr = 0;
446       *handle_ptr = INVALID_HANDLE_VALUE;
447     }
448     release_ioinfo(fdinfo);
449     wxflag_ptr++; handle_ptr++;
450   }
451   return TRUE;
452 }
453 
454 /* INTERNAL: Set up all file descriptors,
455  * as well as default streams (stdin, stderr and stdout)
456  */
457 void msvcrt_init_io(void)
458 {
459   STARTUPINFOA  si;
460   unsigned int  i;
461   ioinfo        *fdinfo;
462 
463   GetStartupInfoA(&si);
464   if (si.cbReserved2 >= sizeof(unsigned int) && si.lpReserved2 != NULL)
465   {
466     BYTE*       wxflag_ptr;
467     HANDLE*     handle_ptr;
468     unsigned int count;
469 
470     count = *(unsigned*)si.lpReserved2;
471     wxflag_ptr = si.lpReserved2 + sizeof(unsigned);
472     handle_ptr = (HANDLE*)(wxflag_ptr + count);
473 
474     count = min(count, (si.cbReserved2 - sizeof(unsigned)) / (sizeof(HANDLE) + 1));
475     count = min(count, MSVCRT_MAX_FILES);
476     for (i = 0; i < count; i++)
477     {
478       if ((*wxflag_ptr & WX_OPEN) && *handle_ptr != INVALID_HANDLE_VALUE)
479       {
480         fdinfo = get_ioinfo_alloc_fd(i);
481         if(fdinfo != &__badioinfo)
482             msvcrt_set_fd(fdinfo, *handle_ptr, *wxflag_ptr);
483         release_ioinfo(fdinfo);
484       }
485 
486       wxflag_ptr++; handle_ptr++;
487     }
488   }
489 
490   fdinfo = get_ioinfo_alloc_fd(STDIN_FILENO);
491   if (!(fdinfo->wxflag & WX_OPEN) || fdinfo->handle == INVALID_HANDLE_VALUE) {
492     HANDLE h = GetStdHandle(STD_INPUT_HANDLE);
493     DWORD type = GetFileType(h);
494 
495     msvcrt_set_fd(fdinfo, h, WX_OPEN|WX_TEXT|((type&0xf)==FILE_TYPE_CHAR ? WX_TTY : 0)
496             |((type&0xf)==FILE_TYPE_PIPE ? WX_PIPE : 0));
497   }
498   release_ioinfo(fdinfo);
499 
500   fdinfo = get_ioinfo_alloc_fd(STDOUT_FILENO);
501   if (!(fdinfo->wxflag & WX_OPEN) || fdinfo->handle == INVALID_HANDLE_VALUE) {
502     HANDLE h = GetStdHandle(STD_OUTPUT_HANDLE);
503     DWORD type = GetFileType(h);
504 
505     msvcrt_set_fd(fdinfo, h, WX_OPEN|WX_TEXT|((type&0xf)==FILE_TYPE_CHAR ? WX_TTY : 0)
506             |((type&0xf)==FILE_TYPE_PIPE ? WX_PIPE : 0));
507   }
508   release_ioinfo(fdinfo);
509 
510   fdinfo = get_ioinfo_alloc_fd(STDERR_FILENO);
511   if (!(fdinfo->wxflag & WX_OPEN) || fdinfo->handle == INVALID_HANDLE_VALUE) {
512     HANDLE h = GetStdHandle(STD_ERROR_HANDLE);
513     DWORD type = GetFileType(h);
514 
515     msvcrt_set_fd(fdinfo, h, WX_OPEN|WX_TEXT|((type&0xf)==FILE_TYPE_CHAR ? WX_TTY : 0)
516             |((type&0xf)==FILE_TYPE_PIPE ? WX_PIPE : 0));
517   }
518   release_ioinfo(fdinfo);
519 
520   TRACE(":handles (%p)(%p)(%p)\n", get_ioinfo_nolock(STDIN_FILENO)->handle,
521         get_ioinfo_nolock(STDOUT_FILENO)->handle,
522         get_ioinfo_nolock(STDERR_FILENO)->handle);
523 
524   memset(_iob,0,3*sizeof(FILE));
525   for (i = 0; i < 3; i++)
526   {
527     /* FILE structs for stdin/out/err are static and never deleted */
528     _iob[i]._file = i;
529     _iob[i]._tmpfname = NULL;
530     _iob[i]._flag = (i == 0) ? _IOREAD : _IOWRT;
531   }
532   MSVCRT_stream_idx = 3;
533 }
534 
535 /* INTERNAL: Flush stdio file buffer */
536 static int msvcrt_flush_buffer(FILE* file)
537 {
538     if((file->_flag & (_IOREAD|_IOWRT)) == _IOWRT &&
539             file->_flag & (_IOMYBUF|_USERBUF)) {
540         int cnt=file->_ptr-file->_base;
541         if(cnt>0 && _write(file->_file, file->_base, cnt) != cnt) {
542             file->_flag |= _IOERR;
543             return EOF;
544         }
545 
546         if(file->_flag & _IORW)
547             file->_flag &= ~_IOWRT;
548 
549 #ifdef __REACTOS__ /* CORE-11949 */
550     file->_ptr=file->_base;
551     file->_cnt=0;
552 #endif
553     }
554 
555 #ifndef __REACTOS__ /* CORE-11949 */
556     file->_ptr=file->_base;
557     file->_cnt=0;
558 #endif
559     return 0;
560 }
561 
562 /*********************************************************************
563  *		_isatty (MSVCRT.@)
564  */
565 int CDECL _isatty(int fd)
566 {
567     TRACE(":fd (%d)\n",fd);
568 
569     return get_ioinfo_nolock(fd)->wxflag & WX_TTY;
570 }
571 
572 /* INTERNAL: Allocate stdio file buffer */
573 /*static*/ BOOL msvcrt_alloc_buffer(FILE* file)
574 {
575     if((file->_file==STDOUT_FILENO || file->_file==STDERR_FILENO)
576             && _isatty(file->_file))
577         return FALSE;
578 
579     file->_base = calloc(1, MSVCRT_INTERNAL_BUFSIZ);
580     if(file->_base) {
581         file->_bufsiz = MSVCRT_INTERNAL_BUFSIZ;
582         file->_flag |= _IOMYBUF;
583     } else {
584         file->_base = (char*)(&file->_charbuf);
585         file->_bufsiz = 2;
586         file->_flag |= _IONBF;
587     }
588     file->_ptr = file->_base;
589     file->_cnt = 0;
590     return TRUE;
591 }
592 
593 /* INTERNAL: Allocate temporary buffer for stdout and stderr */
594 static BOOL add_std_buffer(FILE *file)
595 {
596     static char buffers[2][BUFSIZ];
597 
598     if((file->_file!=STDOUT_FILENO && file->_file!=STDERR_FILENO)
599             || (file->_flag & (_IONBF | _IOMYBUF | _USERBUF))
600             || !_isatty(file->_file))
601         return FALSE;
602 
603     file->_ptr = file->_base = buffers[file->_file == STDOUT_FILENO ? 0 : 1];
604     file->_bufsiz = file->_cnt = BUFSIZ;
605     file->_flag |= _USERBUF;
606     return TRUE;
607 }
608 
609 /* INTERNAL: Removes temporary buffer from stdout or stderr */
610 /* Only call this function when add_std_buffer returned TRUE */
611 static void remove_std_buffer(FILE *file)
612 {
613     msvcrt_flush_buffer(file);
614     file->_ptr = file->_base = NULL;
615     file->_bufsiz = file->_cnt = 0;
616     file->_flag &= ~_USERBUF;
617 }
618 
619 /* INTERNAL: Convert integer to base32 string (0-9a-v), 0 becomes "" */
620 static int msvcrt_int_to_base32(int num, char *str)
621 {
622   char *p;
623   int n = num;
624   int digits = 0;
625 
626   while (n != 0)
627   {
628     n >>= 5;
629     digits++;
630   }
631   p = str + digits;
632   *p = 0;
633   while (--p >= str)
634   {
635     *p = (num & 31) + '0';
636     if (*p > '9')
637       *p += ('a' - '0' - 10);
638     num >>= 5;
639   }
640 
641   return digits;
642 }
643 
644 /* INTERNAL: wide character version of msvcrt_int_to_base32 */
645 static int msvcrt_int_to_base32_w(int num, wchar_t *str)
646 {
647     wchar_t *p;
648     int n = num;
649     int digits = 0;
650 
651     while (n != 0)
652     {
653         n >>= 5;
654         digits++;
655     }
656     p = str + digits;
657     *p = 0;
658     while (--p >= str)
659     {
660         *p = (num & 31) + '0';
661         if (*p > '9')
662             *p += ('a' - '0' - 10);
663         num >>= 5;
664     }
665 
666     return digits;
667 }
668 
669 /* INTERNAL: Create a wide string from an ascii string */
670 wchar_t *msvcrt_wstrdupa(const char *str)
671 {
672   const unsigned int len = strlen(str) + 1 ;
673   wchar_t *wstr = malloc(len* sizeof (wchar_t));
674   if (!wstr)
675     return NULL;
676    MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED,str,len,wstr,len);
677   return wstr;
678 }
679 
680 /*********************************************************************
681  *		__iob_func(MSVCRT.@)
682  */
683 FILE * CDECL __iob_func(void)
684 {
685  return &_iob[0];
686 }
687 
688 /*********************************************************************
689  *		_access (MSVCRT.@)
690  */
691 int CDECL _access(const char *filename, int mode)
692 {
693   DWORD attr = GetFileAttributesA(filename);
694 
695   TRACE("(%s,%d) %d\n",filename,mode,attr);
696 
697   if (!filename || attr == INVALID_FILE_ATTRIBUTES)
698   {
699     _dosmaperr(GetLastError());
700     return -1;
701   }
702   if ((attr & FILE_ATTRIBUTE_READONLY) && (mode & W_OK))
703   {
704     _set_errno(ERROR_ACCESS_DENIED);
705     return -1;
706   }
707   return 0;
708 }
709 
710 /*********************************************************************
711  *		_access_s (MSVCRT.@)
712  */
713 int CDECL _access_s(const char *filename, int mode)
714 {
715   if (!MSVCRT_CHECK_PMT(filename != NULL) ||
716       !MSVCRT_CHECK_PMT((mode & ~(R_OK | W_OK)) == 0))
717   {
718      _set_errno(EINVAL);
719      return -1;
720   }
721 
722   return _access(filename, mode);
723 }
724 
725 /*********************************************************************
726  *		_waccess (MSVCRT.@)
727  */
728 int CDECL _waccess(const wchar_t *filename, int mode)
729 {
730   DWORD attr = GetFileAttributesW(filename);
731 
732   TRACE("(%s,%d) %d\n",debugstr_w(filename),mode,attr);
733 
734   if (!filename || attr == INVALID_FILE_ATTRIBUTES)
735   {
736     _dosmaperr(GetLastError());
737     return -1;
738   }
739   if ((attr & FILE_ATTRIBUTE_READONLY) && (mode & W_OK))
740   {
741     _set_errno(ERROR_ACCESS_DENIED);
742     return -1;
743   }
744   return 0;
745 }
746 
747 /*********************************************************************
748  *		_waccess_s (MSVCRT.@)
749  */
750 int CDECL _waccess_s(const wchar_t *filename, int mode)
751 {
752   if (!MSVCRT_CHECK_PMT(filename != NULL) ||
753       !MSVCRT_CHECK_PMT((mode & ~(R_OK | W_OK)) == 0))
754   {
755      *_errno() = EINVAL;
756      return -1;
757   }
758 
759   return _waccess(filename, mode);
760 }
761 
762 /*********************************************************************
763  *		_chmod (MSVCRT.@)
764  */
765 int CDECL _chmod(const char *path, int flags)
766 {
767   DWORD oldFlags = GetFileAttributesA(path);
768 
769   if (oldFlags != INVALID_FILE_ATTRIBUTES)
770   {
771     DWORD newFlags = (flags & _S_IWRITE)? oldFlags & ~FILE_ATTRIBUTE_READONLY:
772       oldFlags | FILE_ATTRIBUTE_READONLY;
773 
774     if (newFlags == oldFlags || SetFileAttributesA(path, newFlags))
775       return 0;
776   }
777   _dosmaperr(GetLastError());
778   return -1;
779 }
780 
781 /*********************************************************************
782  *		_wchmod (MSVCRT.@)
783  */
784 int CDECL _wchmod(const wchar_t *path, int flags)
785 {
786   DWORD oldFlags = GetFileAttributesW(path);
787 
788   if (oldFlags != INVALID_FILE_ATTRIBUTES)
789   {
790     DWORD newFlags = (flags & _S_IWRITE)? oldFlags & ~FILE_ATTRIBUTE_READONLY:
791       oldFlags | FILE_ATTRIBUTE_READONLY;
792 
793     if (newFlags == oldFlags || SetFileAttributesW(path, newFlags))
794       return 0;
795   }
796   _dosmaperr(GetLastError());
797   return -1;
798 }
799 
800 /*********************************************************************
801  *		_unlink (MSVCRT.@)
802  */
803 int CDECL _unlink(const char *path)
804 {
805   TRACE("%s\n",debugstr_a(path));
806   if(DeleteFileA(path))
807     return 0;
808   TRACE("failed (%d)\n",GetLastError());
809   _dosmaperr(GetLastError());
810   return -1;
811 }
812 
813 /*********************************************************************
814  *		_wunlink (MSVCRT.@)
815  */
816 int CDECL _wunlink(const wchar_t *path)
817 {
818   TRACE("(%s)\n",debugstr_w(path));
819   if(DeleteFileW(path))
820     return 0;
821   TRACE("failed (%d)\n",GetLastError());
822   _dosmaperr(GetLastError());
823   return -1;
824 }
825 
826 /*********************************************************************
827  *		_commit (MSVCRT.@)
828  */
829 int CDECL _commit(int fd)
830 {
831     ioinfo *info = get_ioinfo(fd);
832     int ret;
833 
834     TRACE(":fd (%d) handle (%p)\n", fd, info->handle);
835     if (info->handle == INVALID_HANDLE_VALUE)
836         ret = -1;
837     else if (!FlushFileBuffers(info->handle))
838     {
839         if (GetLastError() == ERROR_INVALID_HANDLE)
840         {
841           /* FlushFileBuffers fails for console handles
842            * so we ignore this error.
843            */
844           ret = 0;
845         }
846         else
847         {
848             TRACE(":failed-last error (%d)\n",GetLastError());
849             _dosmaperr(GetLastError());
850             ret = -1;
851         }
852     }
853     else
854     {
855         TRACE(":ok\n");
856         ret = 0;
857     }
858 
859     release_ioinfo(info);
860     return ret;
861 }
862 
863 /* _flushall calls fflush which calls _flushall */
864 int CDECL fflush(FILE* file);
865 
866 /* INTERNAL: Flush all stream buffer */
867 static int msvcrt_flush_all_buffers(int mask)
868 {
869   int i, num_flushed = 0;
870   FILE *file;
871 
872   LOCK_FILES();
873   for (i = 0; i < MSVCRT_stream_idx; i++) {
874     file = msvcrt_get_file(i);
875 
876     if (file->_flag)
877     {
878       if(file->_flag & mask) {
879 	fflush(file);
880         num_flushed++;
881       }
882     }
883   }
884   UNLOCK_FILES();
885 
886   TRACE(":flushed (%d) handles\n",num_flushed);
887   return num_flushed;
888 }
889 
890 /*********************************************************************
891  *		_flushall (MSVCRT.@)
892  */
893 int CDECL _flushall(void)
894 {
895     return msvcrt_flush_all_buffers(_IOWRT | _IOREAD);
896 }
897 
898 /*********************************************************************
899  *		fflush (MSVCRT.@)
900  */
901 int CDECL fflush(FILE* file)
902 {
903     if(!file) {
904         msvcrt_flush_all_buffers(_IOWRT);
905     } else if(file->_flag & _IOWRT) {
906         int res;
907 
908         _lock_file(file);
909         res = msvcrt_flush_buffer(file);
910         /* FIXME
911         if(!res && (file->_flag & _IOCOMMIT))
912             res = _commit(file->_file) ? EOF : 0;
913         */
914         _unlock_file(file);
915 
916         return res;
917     } else if(file->_flag & _IOREAD) {
918         _lock_file(file);
919         file->_cnt = 0;
920         file->_ptr = file->_base;
921         _unlock_file(file);
922 
923         return 0;
924     }
925     return 0;
926 }
927 
928 /*********************************************************************
929  *		_close (MSVCRT.@)
930  */
931 int CDECL _close(int fd)
932 {
933   ioinfo *info = get_ioinfo(fd);
934   int ret;
935 
936   TRACE(":fd (%d) handle (%p)\n", fd, info->handle);
937   if (!(info->wxflag & WX_OPEN)) {
938     ret = -1;
939   } else if (fd == STDOUT_FILENO &&
940           info->handle == get_ioinfo_nolock(STDERR_FILENO)->handle) {
941     msvcrt_free_fd(fd);
942     ret = 0;
943   } else if (fd == STDERR_FILENO &&
944           info->handle == get_ioinfo_nolock(STDOUT_FILENO)->handle) {
945     msvcrt_free_fd(fd);
946     ret = 0;
947   } else {
948     ret = CloseHandle(info->handle) ? 0 : -1;
949     msvcrt_free_fd(fd);
950     if (ret) {
951       WARN(":failed-last error (%d)\n",GetLastError());
952       _dosmaperr(GetLastError());
953       ret = -1;
954     }
955   }
956   release_ioinfo(info);
957   return ret;
958 }
959 
960 /*********************************************************************
961  *		_dup2 (MSVCRT.@)
962  * NOTES
963  * MSDN isn't clear on this point, but the remarks for _pipe
964  * indicate file descriptors duplicated with _dup and _dup2 are always
965  * inheritable.
966  */
967 int CDECL _dup2(int od, int nd)
968 {
969   ioinfo *info_od, *info_nd;
970   int ret;
971 
972   TRACE("(od=%d, nd=%d)\n", od, nd);
973 
974   if (od < nd)
975   {
976     info_od = get_ioinfo(od);
977     info_nd = get_ioinfo_alloc_fd(nd);
978   }
979   else
980   {
981     info_nd = get_ioinfo_alloc_fd(nd);
982     info_od = get_ioinfo(od);
983   }
984 
985   if (info_nd == &__badioinfo)
986   {
987       ret = -1;
988   }
989   else if (info_od->wxflag & WX_OPEN)
990   {
991     HANDLE handle;
992 
993     if (DuplicateHandle(GetCurrentProcess(), info_od->handle,
994      GetCurrentProcess(), &handle, 0, TRUE, DUPLICATE_SAME_ACCESS))
995     {
996       int wxflag = info_od->wxflag & ~_O_NOINHERIT;
997 
998       if (info_nd->wxflag & WX_OPEN)
999         _close(nd);
1000 
1001       msvcrt_set_fd(info_nd, handle, wxflag);
1002       /* _dup2 returns 0, not nd, on success */
1003       ret = 0;
1004     }
1005     else
1006     {
1007       ret = -1;
1008       _dosmaperr(GetLastError());
1009     }
1010   }
1011   else
1012   {
1013     *_errno() = EBADF;
1014     ret = -1;
1015   }
1016 
1017   release_ioinfo(info_od);
1018   release_ioinfo(info_nd);
1019   return ret;
1020 }
1021 
1022 /*********************************************************************
1023  *		_dup (MSVCRT.@)
1024  */
1025 int CDECL _dup(int od)
1026 {
1027   int fd, ret;
1028   ioinfo *info = get_ioinfo_alloc(&fd);
1029 
1030   if (_dup2(od, fd) == 0)
1031     ret = fd;
1032   else
1033     ret = -1;
1034   release_ioinfo(info);
1035   return ret;
1036 }
1037 
1038 /*********************************************************************
1039  *		_eof (MSVCRT.@)
1040  */
1041 int CDECL _eof(int fd)
1042 {
1043   ioinfo *info = get_ioinfo(fd);
1044   DWORD curpos,endpos;
1045   LONG hcurpos,hendpos;
1046 
1047   TRACE(":fd (%d) handle (%p)\n", fd, info->handle);
1048 
1049   if (info->handle == INVALID_HANDLE_VALUE)
1050   {
1051     release_ioinfo(info);
1052     return -1;
1053   }
1054 
1055   if (info->wxflag & WX_ATEOF)
1056   {
1057       release_ioinfo(info);
1058       return TRUE;
1059   }
1060 
1061   /* Otherwise we do it the hard way */
1062   hcurpos = hendpos = 0;
1063   curpos = SetFilePointer(info->handle, 0, &hcurpos, FILE_CURRENT);
1064   endpos = SetFilePointer(info->handle, 0, &hendpos, FILE_END);
1065 
1066   if (curpos == endpos && hcurpos == hendpos)
1067   {
1068     /* FIXME: shouldn't WX_ATEOF be set here? */
1069     release_ioinfo(info);
1070     return TRUE;
1071   }
1072 
1073   SetFilePointer(info->handle, curpos, &hcurpos, FILE_BEGIN);
1074   release_ioinfo(info);
1075   return FALSE;
1076 }
1077 
1078 /*********************************************************************
1079  *		_fcloseall (MSVCRT.@)
1080  */
1081 int CDECL _fcloseall(void)
1082 {
1083   int num_closed = 0, i;
1084   FILE *file;
1085 
1086   LOCK_FILES();
1087   for (i = 3; i < MSVCRT_stream_idx; i++) {
1088     file = msvcrt_get_file(i);
1089 
1090     if (file->_flag && !fclose(file))
1091       num_closed++;
1092   }
1093   UNLOCK_FILES();
1094 
1095   TRACE(":closed (%d) handles\n",num_closed);
1096   return num_closed;
1097 }
1098 
1099 /* free everything on process exit */
1100 void msvcrt_free_io(void)
1101 {
1102     unsigned int i;
1103     int j;
1104 
1105     _flushall();
1106     _fcloseall();
1107 
1108     for(i=0; i<sizeof(__pioinfo)/sizeof(__pioinfo[0]); i++)
1109     {
1110         if(!__pioinfo[i])
1111             continue;
1112 
1113         for(j=0; j<MSVCRT_FD_BLOCK_SIZE; j++)
1114         {
1115             if(__pioinfo[i][j].exflag & EF_CRIT_INIT)
1116                 DeleteCriticalSection(&__pioinfo[i][j].crit);
1117         }
1118         free(__pioinfo[i]);
1119     }
1120 
1121     for(j=0; j<MSVCRT_stream_idx; j++)
1122     {
1123         FILE *file = msvcrt_get_file(j);
1124         if(file<_iob || file>=_iob+_IOB_ENTRIES)
1125         {
1126             ((file_crit*)file)->crit.DebugInfo->Spare[0] = 0;
1127             DeleteCriticalSection(&((file_crit*)file)->crit);
1128         }
1129     }
1130 
1131     for(i=0; i<sizeof(MSVCRT_fstream)/sizeof(MSVCRT_fstream[0]); i++)
1132         free(MSVCRT_fstream[i]);
1133 }
1134 
1135 /*********************************************************************
1136  *		_lseeki64 (MSVCRT.@)
1137  */
1138 __int64 CDECL _lseeki64(int fd, __int64 offset, int whence)
1139 {
1140   ioinfo *info = get_ioinfo(fd);
1141   LARGE_INTEGER ofs;
1142 
1143   TRACE(":fd (%d) handle (%p)\n", fd, info->handle);
1144 
1145   if (info->handle == INVALID_HANDLE_VALUE)
1146   {
1147     release_ioinfo(info);
1148     return -1;
1149   }
1150 
1151   if (whence < 0 || whence > 2)
1152   {
1153     release_ioinfo(info);
1154     *_errno() = EINVAL;
1155     return -1;
1156   }
1157 
1158   TRACE(":fd (%d) to %s pos %s\n",
1159         fd,wine_dbgstr_longlong(offset),
1160         (whence==SEEK_SET)?"SEEK_SET":
1161         (whence==SEEK_CUR)?"SEEK_CUR":
1162         (whence==SEEK_END)?"SEEK_END":"UNKNOWN");
1163 
1164   /* The MoleBox protection scheme expects msvcrt to use SetFilePointer only,
1165    * so a LARGE_INTEGER offset cannot be passed directly via SetFilePointerEx. */
1166   ofs.QuadPart = offset;
1167   if ((ofs.u.LowPart = SetFilePointer(info->handle, ofs.u.LowPart, &ofs.u.HighPart, whence)) != INVALID_SET_FILE_POINTER ||
1168       GetLastError() == ERROR_SUCCESS)
1169   {
1170     info->wxflag &= ~(WX_ATEOF|WX_READEOF);
1171     /* FIXME: What if we seek _to_ EOF - is EOF set? */
1172 
1173     release_ioinfo(info);
1174     return ofs.QuadPart;
1175   }
1176   release_ioinfo(info);
1177   TRACE(":error-last error (%d)\n",GetLastError());
1178   _dosmaperr(GetLastError());
1179   return -1;
1180 }
1181 
1182 /*********************************************************************
1183  *		_lseek (MSVCRT.@)
1184  */
1185 LONG CDECL _lseek(int fd, LONG offset, int whence)
1186 {
1187     return (LONG)_lseeki64(fd, offset, whence);
1188 }
1189 
1190 /*********************************************************************
1191  *              _lock_file (MSVCRT.@)
1192  */
1193 void CDECL _lock_file(FILE *file)
1194 {
1195     if(file>=_iob && file<_iob+_IOB_ENTRIES)
1196         _lock(_STREAM_LOCKS+(file-_iob));
1197     /* ReactOS: string streams dont need to be locked */
1198     else if(!(file->_flag & _IOSTRG))
1199         EnterCriticalSection(&((file_crit*)file)->crit);
1200 }
1201 
1202 /*********************************************************************
1203  *              _unlock_file (MSVCRT.@)
1204  */
1205 void CDECL _unlock_file(FILE *file)
1206 {
1207     if(file>=_iob && file<_iob+_IOB_ENTRIES)
1208         _unlock(_STREAM_LOCKS+(file-_iob));
1209     /* ReactOS: string streams dont need to be locked */
1210     else if(!(file->_flag & _IOSTRG))
1211         LeaveCriticalSection(&((file_crit*)file)->crit);
1212 
1213 }
1214 
1215 /*********************************************************************
1216  *		_locking (MSVCRT.@)
1217  *
1218  * This is untested; the underlying LockFile doesn't work yet.
1219  */
1220 int CDECL _locking(int fd, int mode, LONG nbytes)
1221 {
1222   ioinfo *info = get_ioinfo(fd);
1223   BOOL ret;
1224   DWORD cur_locn;
1225 
1226   TRACE(":fd (%d) handle (%p)\n", fd, info->handle);
1227   if (info->handle == INVALID_HANDLE_VALUE)
1228   {
1229     release_ioinfo(info);
1230     return -1;
1231   }
1232 
1233   if (mode < 0 || mode > 4)
1234   {
1235     release_ioinfo(info);
1236     *_errno() = EINVAL;
1237     return -1;
1238   }
1239 
1240   TRACE(":fd (%d) by 0x%08x mode %s\n",
1241         fd,nbytes,(mode==_LK_UNLCK)?"_LK_UNLCK":
1242         (mode==_LK_LOCK)?"_LK_LOCK":
1243         (mode==_LK_NBLCK)?"_LK_NBLCK":
1244         (mode==_LK_RLCK)?"_LK_RLCK":
1245         (mode==_LK_NBRLCK)?"_LK_NBRLCK":
1246                           "UNKNOWN");
1247 
1248   if ((cur_locn = SetFilePointer(info->handle, 0L, NULL, FILE_CURRENT)) == INVALID_SET_FILE_POINTER)
1249   {
1250     release_ioinfo(info);
1251     FIXME ("Seek failed\n");
1252     *_errno() = EINVAL; /* FIXME */
1253     return -1;
1254   }
1255   if (mode == _LK_LOCK || mode == _LK_RLCK)
1256   {
1257     int nretry = 10;
1258     ret = 1; /* just to satisfy gcc */
1259     while (nretry--)
1260     {
1261       ret = LockFile(info->handle, cur_locn, 0L, nbytes, 0L);
1262       if (ret) break;
1263       Sleep(1);
1264     }
1265   }
1266   else if (mode == _LK_UNLCK)
1267     ret = UnlockFile(info->handle, cur_locn, 0L, nbytes, 0L);
1268   else
1269     ret = LockFile(info->handle, cur_locn, 0L, nbytes, 0L);
1270   /* FIXME - what about error settings? */
1271   release_ioinfo(info);
1272   return ret ? 0 : -1;
1273 }
1274 
1275 /*********************************************************************
1276  *		_fseeki64 (MSVCRT.@)
1277  */
1278 int CDECL _fseeki64(FILE* file, __int64 offset, int whence)
1279 {
1280   int ret;
1281 
1282   _lock_file(file);
1283   /* Flush output if needed */
1284   if(file->_flag & _IOWRT)
1285 	msvcrt_flush_buffer(file);
1286 
1287   if(whence == SEEK_CUR && file->_flag & _IOREAD ) {
1288       whence = SEEK_SET;
1289       offset += _ftelli64(file);
1290   }
1291 
1292   /* Discard buffered input */
1293   file->_cnt = 0;
1294   file->_ptr = file->_base;
1295   /* Reset direction of i/o */
1296   if(file->_flag & _IORW) {
1297         file->_flag &= ~(_IOREAD|_IOWRT);
1298   }
1299   /* Clear end of file flag */
1300   file->_flag &= ~_IOEOF;
1301   ret = (_lseeki64(file->_file,offset,whence) == -1)?-1:0;
1302 
1303   _unlock_file(file);
1304   return ret;
1305 }
1306 
1307 /*********************************************************************
1308  *		fseek (MSVCRT.@)
1309  */
1310 int CDECL fseek(FILE* file, long offset, int whence)
1311 {
1312     return _fseeki64( file, offset, whence );
1313 }
1314 
1315 /*********************************************************************
1316  *		_chsize_s (MSVCRT.@)
1317  */
1318 int CDECL _chsize_s(int fd, __int64 size)
1319 {
1320     ioinfo *info;
1321     __int64 cur, pos;
1322     BOOL ret = FALSE;
1323 
1324     TRACE("(fd=%d, size=%s)\n", fd, wine_dbgstr_longlong(size));
1325 
1326     if (!MSVCRT_CHECK_PMT(size >= 0)) return EINVAL;
1327 
1328     info = get_ioinfo(fd);
1329     if (info->handle != INVALID_HANDLE_VALUE)
1330     {
1331         /* save the current file pointer */
1332         cur = _lseeki64(fd, 0, SEEK_CUR);
1333         if (cur >= 0)
1334         {
1335             pos = _lseeki64(fd, size, SEEK_SET);
1336             if (pos >= 0)
1337             {
1338                 ret = SetEndOfFile(info->handle);
1339                 if (!ret) _dosmaperr(GetLastError());
1340             }
1341 
1342             /* restore the file pointer */
1343             _lseeki64(fd, cur, SEEK_SET);
1344         }
1345     }
1346 
1347     release_ioinfo(info);
1348     return ret ? 0 : *_errno();
1349 }
1350 
1351 /*********************************************************************
1352  *		_chsize (MSVCRT.@)
1353  */
1354 int CDECL _chsize(int fd, long size)
1355 {
1356     /* _chsize_s returns errno on failure but _chsize should return -1 */
1357     return _chsize_s( fd, size ) == 0 ? 0 : -1;
1358 }
1359 
1360 /*********************************************************************
1361  *		clearerr (MSVCRT.@)
1362  */
1363 void CDECL clearerr(FILE* file)
1364 {
1365   TRACE(":file (%p) fd (%d)\n",file,file->_file);
1366 
1367   _lock_file(file);
1368   file->_flag &= ~(_IOERR | _IOEOF);
1369   _unlock_file(file);
1370 }
1371 
1372 /*********************************************************************
1373  *		rewind (MSVCRT.@)
1374  */
1375 void CDECL rewind(FILE* file)
1376 {
1377   TRACE(":file (%p) fd (%d)\n",file,file->_file);
1378 
1379   _lock_file(file);
1380   fseek(file, 0L, SEEK_SET);
1381   clearerr(file);
1382   _unlock_file(file);
1383 }
1384 
1385 static int msvcrt_get_flags(const wchar_t* mode, int *open_flags, int* stream_flags)
1386 {
1387   int plus = strchrW(mode, '+') != NULL;
1388 
1389   TRACE("%s\n", debugstr_w(mode));
1390 
1391   while(*mode == ' ') mode++;
1392 
1393   switch(*mode++)
1394   {
1395   case 'R': case 'r':
1396     *open_flags = plus ? _O_RDWR : _O_RDONLY;
1397     *stream_flags = plus ? _IORW : _IOREAD;
1398     break;
1399   case 'W': case 'w':
1400     *open_flags = _O_CREAT | _O_TRUNC | (plus  ? _O_RDWR : _O_WRONLY);
1401     *stream_flags = plus ? _IORW : _IOWRT;
1402     break;
1403   case 'A': case 'a':
1404     *open_flags = _O_CREAT | _O_APPEND | (plus  ? _O_RDWR : _O_WRONLY);
1405     *stream_flags = plus ? _IORW : _IOWRT;
1406     break;
1407   default:
1408     MSVCRT_INVALID_PMT(0, EINVAL);
1409     return -1;
1410   }
1411 
1412   *stream_flags |= _commode;
1413 
1414   while (*mode && *mode!=',')
1415     switch (*mode++)
1416     {
1417     case 'B': case 'b':
1418       *open_flags |=  _O_BINARY;
1419       *open_flags &= ~_O_TEXT;
1420       break;
1421     case 't':
1422       *open_flags |=  _O_TEXT;
1423       *open_flags &= ~_O_BINARY;
1424       break;
1425     case 'D':
1426       *open_flags |= _O_TEMPORARY;
1427       break;
1428     case 'T':
1429       *open_flags |= _O_SHORT_LIVED;
1430       break;
1431     case 'c':
1432       *stream_flags |= _IOCOMMIT;
1433       break;
1434     case 'n':
1435       *stream_flags &= ~_IOCOMMIT;
1436       break;
1437     case 'N':
1438       *open_flags |= _O_NOINHERIT;
1439       break;
1440     case '+':
1441     case ' ':
1442     case 'a':
1443     case 'w':
1444       break;
1445     case 'S':
1446     case 'R':
1447       FIXME("ignoring cache optimization flag: %c\n", mode[-1]);
1448       break;
1449     default:
1450       ERR("incorrect mode flag: %c\n", mode[-1]);
1451       break;
1452     }
1453 
1454   if(*mode == ',')
1455   {
1456     static const WCHAR ccs[] = {'c','c','s'};
1457     static const WCHAR utf8[] = {'u','t','f','-','8'};
1458     static const WCHAR utf16le[] = {'u','t','f','-','1','6','l','e'};
1459     static const WCHAR unicode[] = {'u','n','i','c','o','d','e'};
1460 
1461     mode++;
1462     while(*mode == ' ') mode++;
1463     if(!MSVCRT_CHECK_PMT(!strncmpW(ccs, mode, sizeof(ccs)/sizeof(ccs[0]))))
1464       return -1;
1465     mode += sizeof(ccs)/sizeof(ccs[0]);
1466     while(*mode == ' ') mode++;
1467     if(!MSVCRT_CHECK_PMT(*mode == '='))
1468         return -1;
1469     mode++;
1470     while(*mode == ' ') mode++;
1471 
1472     if(!strncmpiW(utf8, mode, sizeof(utf8)/sizeof(utf8[0])))
1473     {
1474       *open_flags |= _O_U8TEXT;
1475       mode += sizeof(utf8)/sizeof(utf8[0]);
1476     }
1477     else if(!strncmpiW(utf16le, mode, sizeof(utf16le)/sizeof(utf16le[0])))
1478     {
1479       *open_flags |= _O_U16TEXT;
1480       mode += sizeof(utf16le)/sizeof(utf16le[0]);
1481     }
1482     else if(!strncmpiW(unicode, mode, sizeof(unicode)/sizeof(unicode[0])))
1483     {
1484       *open_flags |= _O_WTEXT;
1485       mode += sizeof(unicode)/sizeof(unicode[0]);
1486     }
1487     else
1488     {
1489       MSVCRT_INVALID_PMT(0, EINVAL);
1490       return -1;
1491     }
1492 
1493     while(*mode == ' ') mode++;
1494   }
1495 
1496   if(!MSVCRT_CHECK_PMT(*mode == 0))
1497     return -1;
1498   return 0;
1499 }
1500 
1501 /*********************************************************************
1502  *		_fdopen (MSVCRT.@)
1503  */
1504 FILE* CDECL _fdopen(int fd, const char *mode)
1505 {
1506     FILE *ret;
1507     wchar_t *modeW = NULL;
1508 
1509     if (mode && !(modeW = msvcrt_wstrdupa(mode))) return NULL;
1510 
1511     ret = _wfdopen(fd, modeW);
1512 
1513     free(modeW);
1514     return ret;
1515 }
1516 
1517 /*********************************************************************
1518  *		_wfdopen (MSVCRT.@)
1519  */
1520 FILE* CDECL _wfdopen(int fd, const wchar_t *mode)
1521 {
1522   int open_flags, stream_flags;
1523   FILE* file;
1524 
1525   if (msvcrt_get_flags(mode, &open_flags, &stream_flags) == -1) return NULL;
1526 
1527   LOCK_FILES();
1528   if (!(file = msvcrt_alloc_fp()))
1529     file = NULL;
1530   else if (msvcrt_init_fp(file, fd, stream_flags) == -1)
1531   {
1532     file->_flag = 0;
1533     file = NULL;
1534   }
1535   else TRACE(":fd (%d) mode (%s) FILE* (%p)\n", fd, debugstr_w(mode), file);
1536   UNLOCK_FILES();
1537 
1538   return file;
1539 }
1540 
1541 /*********************************************************************
1542  *		_filelength (MSVCRT.@)
1543  */
1544 LONG CDECL _filelength(int fd)
1545 {
1546   LONG curPos = _lseek(fd, 0, SEEK_CUR);
1547   if (curPos != -1)
1548   {
1549     LONG endPos = _lseek(fd, 0, SEEK_END);
1550     if (endPos != -1)
1551     {
1552       if (endPos != curPos)
1553         _lseek(fd, curPos, SEEK_SET);
1554       return endPos;
1555     }
1556   }
1557   return -1;
1558 }
1559 
1560 /*********************************************************************
1561  *		_filelengthi64 (MSVCRT.@)
1562  */
1563 __int64 CDECL _filelengthi64(int fd)
1564 {
1565   __int64 curPos = _lseeki64(fd, 0, SEEK_CUR);
1566   if (curPos != -1)
1567   {
1568     __int64 endPos = _lseeki64(fd, 0, SEEK_END);
1569     if (endPos != -1)
1570     {
1571       if (endPos != curPos)
1572         _lseeki64(fd, curPos, SEEK_SET);
1573       return endPos;
1574     }
1575   }
1576   return -1;
1577 }
1578 
1579 /*********************************************************************
1580  *		_fileno (MSVCRT.@)
1581  */
1582 int CDECL _fileno(FILE* file)
1583 {
1584   TRACE(":FILE* (%p) fd (%d)\n",file,file->_file);
1585   return file->_file;
1586 }
1587 
1588 /*********************************************************************
1589  *		_get_osfhandle (MSVCRT.@)
1590  */
1591 intptr_t CDECL _get_osfhandle(int fd)
1592 {
1593   HANDLE hand = get_ioinfo_nolock(fd)->handle;
1594   TRACE(":fd (%d) handle (%p)\n",fd,hand);
1595 
1596   if(hand == INVALID_HANDLE_VALUE)
1597       *_errno() = EBADF;
1598   return (intptr_t)hand;
1599 }
1600 
1601 /*********************************************************************
1602  *		_mktemp (MSVCRT.@)
1603  */
1604 char * CDECL _mktemp(char *pattern)
1605 {
1606   int numX = 0;
1607   char *retVal = pattern;
1608   int id;
1609   char letter = 'a';
1610 
1611   if(!pattern)
1612       return NULL;
1613 
1614   while(*pattern)
1615     numX = (*pattern++ == 'X')? numX + 1 : 0;
1616   if (numX < 6)
1617     return NULL;
1618   pattern--;
1619   id = GetCurrentProcessId();
1620   numX = 6;
1621   while(numX--)
1622   {
1623     int tempNum = id / 10;
1624     *pattern-- = id - (tempNum * 10) + '0';
1625     id = tempNum;
1626   }
1627   pattern++;
1628   do
1629   {
1630     *pattern = letter++;
1631     if (GetFileAttributesA(retVal) == INVALID_FILE_ATTRIBUTES)
1632       return retVal;
1633   } while(letter <= 'z');
1634   return NULL;
1635 }
1636 
1637 /*********************************************************************
1638  *		_wmktemp (MSVCRT.@)
1639  */
1640 wchar_t * CDECL _wmktemp(wchar_t *pattern)
1641 {
1642   int numX = 0;
1643   wchar_t *retVal = pattern;
1644   int id;
1645   wchar_t letter = 'a';
1646 
1647   while(*pattern)
1648     numX = (*pattern++ == 'X')? numX + 1 : 0;
1649   if (numX < 5)
1650     return NULL;
1651   pattern--;
1652   id = GetCurrentProcessId();
1653   numX = 6;
1654   while(numX--)
1655   {
1656     int tempNum = id / 10;
1657     *pattern-- = id - (tempNum * 10) + '0';
1658     id = tempNum;
1659   }
1660   pattern++;
1661   do
1662   {
1663     if (GetFileAttributesW(retVal) == INVALID_FILE_ATTRIBUTES &&
1664         GetLastError() == ERROR_FILE_NOT_FOUND)
1665       return retVal;
1666     *pattern = letter++;
1667   } while(letter != '|');
1668   return NULL;
1669 }
1670 
1671 /*static*/ unsigned split_oflags(unsigned oflags)
1672 {
1673     int         wxflags = 0;
1674     unsigned unsupp; /* until we support everything */
1675 
1676     if (oflags & _O_APPEND)              wxflags |= WX_APPEND;
1677     if (oflags & _O_BINARY)              {/* Nothing to do */}
1678     else if (oflags & _O_TEXT)           wxflags |= WX_TEXT;
1679     else if (oflags & _O_WTEXT)          wxflags |= WX_TEXT;
1680     else if (oflags & _O_U16TEXT)        wxflags |= WX_TEXT;
1681     else if (oflags & _O_U8TEXT)         wxflags |= WX_TEXT;
1682     else if (*__p__fmode() & _O_BINARY)  {/* Nothing to do */}
1683     else                                        wxflags |= WX_TEXT; /* default to TEXT*/
1684     if (oflags & _O_NOINHERIT)           wxflags |= WX_DONTINHERIT;
1685 
1686     if ((unsupp = oflags & ~(
1687                     _O_BINARY|_O_TEXT|_O_APPEND|
1688                     _O_TRUNC|_O_EXCL|_O_CREAT|
1689                     _O_RDWR|_O_WRONLY|_O_TEMPORARY|
1690                     _O_NOINHERIT|
1691                     _O_SEQUENTIAL|_O_RANDOM|_O_SHORT_LIVED|
1692                     _O_WTEXT|_O_U16TEXT|_O_U8TEXT
1693                     )))
1694         ERR(":unsupported oflags 0x%04x\n",unsupp);
1695 
1696     return wxflags;
1697 }
1698 
1699 /*********************************************************************
1700  *              _pipe (MSVCRT.@)
1701  */
1702 int CDECL _pipe(int *pfds, unsigned int psize, int textmode)
1703 {
1704   int ret = -1;
1705   SECURITY_ATTRIBUTES sa;
1706   HANDLE readHandle, writeHandle;
1707 
1708   if (!pfds)
1709   {
1710     *_errno() = EINVAL;
1711     return -1;
1712   }
1713 
1714   sa.nLength = sizeof(SECURITY_ATTRIBUTES);
1715   sa.bInheritHandle = !(textmode & _O_NOINHERIT);
1716   sa.lpSecurityDescriptor = NULL;
1717   if (CreatePipe(&readHandle, &writeHandle, &sa, psize))
1718   {
1719     unsigned int wxflags = split_oflags(textmode);
1720     int fd;
1721 
1722     fd = msvcrt_alloc_fd(readHandle, wxflags);
1723     if (fd != -1)
1724     {
1725       pfds[0] = fd;
1726       fd = msvcrt_alloc_fd(writeHandle, wxflags);
1727       if (fd != -1)
1728       {
1729         pfds[1] = fd;
1730         ret = 0;
1731       }
1732       else
1733       {
1734         _close(pfds[0]);
1735         CloseHandle(writeHandle);
1736         *_errno() = EMFILE;
1737       }
1738     }
1739     else
1740     {
1741       CloseHandle(readHandle);
1742       CloseHandle(writeHandle);
1743       *_errno() = EMFILE;
1744     }
1745   }
1746   else
1747     _dosmaperr(GetLastError());
1748 
1749   return ret;
1750 }
1751 
1752 static int check_bom(HANDLE h, int oflags, BOOL seek)
1753 {
1754     char bom[sizeof(utf8_bom)];
1755     DWORD r;
1756 
1757     oflags &= ~(_O_WTEXT|_O_U16TEXT|_O_U8TEXT);
1758 
1759     if (!ReadFile(h, bom, sizeof(utf8_bom), &r, NULL))
1760         return oflags;
1761 
1762     if (r==sizeof(utf8_bom) && !memcmp(bom, utf8_bom, sizeof(utf8_bom))) {
1763         oflags |= _O_U8TEXT;
1764     }else if (r>=sizeof(utf16_bom) && !memcmp(bom, utf16_bom, sizeof(utf16_bom))) {
1765         if (seek && r>2)
1766             SetFilePointer(h, 2, NULL, FILE_BEGIN);
1767         oflags |= _O_U16TEXT;
1768     }else if (seek) {
1769         SetFilePointer(h, 0, NULL, FILE_BEGIN);
1770     }
1771 
1772     return oflags;
1773 }
1774 
1775 /*********************************************************************
1776  *              _wsopen_s (MSVCRT.@)
1777  */
1778 int CDECL _wsopen_s( int *fd, const wchar_t* path, int oflags, int shflags, int pmode )
1779 {
1780   DWORD access = 0, creation = 0, attrib;
1781   SECURITY_ATTRIBUTES sa;
1782   DWORD sharing, type;
1783   int wxflag;
1784   HANDLE hand;
1785 
1786   TRACE("fd*: %p :file (%s) oflags: 0x%04x shflags: 0x%04x pmode: 0x%04x\n",
1787         fd, debugstr_w(path), oflags, shflags, pmode);
1788 
1789   if (!MSVCRT_CHECK_PMT( fd != NULL )) return EINVAL;
1790 
1791   *fd = -1;
1792   wxflag = split_oflags(oflags);
1793   switch (oflags & (_O_RDONLY | _O_WRONLY | _O_RDWR))
1794   {
1795   case _O_RDONLY: access |= GENERIC_READ; break;
1796   case _O_WRONLY: access |= GENERIC_WRITE; break;
1797   case _O_RDWR:   access |= GENERIC_WRITE | GENERIC_READ; break;
1798   }
1799 
1800   if (oflags & _O_CREAT)
1801   {
1802     if(pmode & ~(_S_IREAD | _S_IWRITE))
1803       FIXME(": pmode 0x%04x ignored\n", pmode);
1804     else
1805       WARN(": pmode 0x%04x ignored\n", pmode);
1806 
1807     if (oflags & _O_EXCL)
1808       creation = CREATE_NEW;
1809     else if (oflags & _O_TRUNC)
1810       creation = CREATE_ALWAYS;
1811     else
1812       creation = OPEN_ALWAYS;
1813   }
1814   else  /* no _O_CREAT */
1815   {
1816     if (oflags & _O_TRUNC)
1817       creation = TRUNCATE_EXISTING;
1818     else
1819       creation = OPEN_EXISTING;
1820   }
1821 
1822   switch( shflags )
1823   {
1824     case _SH_DENYRW:
1825       sharing = 0L;
1826       break;
1827     case _SH_DENYWR:
1828       sharing = FILE_SHARE_READ;
1829       break;
1830     case _SH_DENYRD:
1831       sharing = FILE_SHARE_WRITE;
1832       break;
1833     case _SH_DENYNO:
1834       sharing = FILE_SHARE_READ | FILE_SHARE_WRITE;
1835       break;
1836     default:
1837       ERR( "Unhandled shflags 0x%x\n", shflags );
1838       return EINVAL;
1839   }
1840   attrib = FILE_ATTRIBUTE_NORMAL;
1841 
1842   if (oflags & _O_TEMPORARY)
1843   {
1844       attrib |= FILE_FLAG_DELETE_ON_CLOSE;
1845       access |= DELETE;
1846       sharing |= FILE_SHARE_DELETE;
1847   }
1848 
1849   sa.nLength              = sizeof( SECURITY_ATTRIBUTES );
1850   sa.lpSecurityDescriptor = NULL;
1851   sa.bInheritHandle       = !(oflags & _O_NOINHERIT);
1852 
1853   if ((oflags&(_O_WTEXT|_O_U16TEXT|_O_U8TEXT))
1854           && (creation==OPEN_ALWAYS || creation==OPEN_EXISTING)
1855           && !(access&GENERIC_READ))
1856   {
1857       hand = CreateFileW(path, GENERIC_READ, FILE_SHARE_READ|FILE_SHARE_WRITE,
1858               &sa, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
1859       if (hand != INVALID_HANDLE_VALUE)
1860       {
1861           oflags = check_bom(hand, oflags, FALSE);
1862           CloseHandle(hand);
1863       }
1864       else
1865           oflags &= ~(_O_WTEXT|_O_U16TEXT|_O_U8TEXT);
1866   }
1867 
1868   hand = CreateFileW(path, access, sharing, &sa, creation, attrib, 0);
1869   if (hand == INVALID_HANDLE_VALUE)  {
1870     WARN(":failed-last error (%d)\n",GetLastError());
1871     _dosmaperr(GetLastError());
1872     return *_errno();
1873   }
1874 
1875   if (oflags & (_O_WTEXT|_O_U16TEXT|_O_U8TEXT))
1876   {
1877       if ((access & GENERIC_WRITE) && (creation==CREATE_NEW
1878                   || creation==CREATE_ALWAYS || creation==TRUNCATE_EXISTING
1879                   || (creation==OPEN_ALWAYS && GetLastError()==ERROR_ALREADY_EXISTS)))
1880       {
1881           if (oflags & _O_U8TEXT)
1882           {
1883               DWORD written = 0, tmp;
1884 
1885               while(written!=sizeof(utf8_bom) && WriteFile(hand, (char*)utf8_bom+written,
1886                           sizeof(utf8_bom)-written, &tmp, NULL))
1887                   written += tmp;
1888               if (written != sizeof(utf8_bom)) {
1889                   WARN("error writing BOM\n");
1890                   CloseHandle(hand);
1891                   _dosmaperr(GetLastError());
1892                   return *_errno();
1893               }
1894           }
1895           else
1896           {
1897               DWORD written = 0, tmp;
1898 
1899               while(written!=sizeof(utf16_bom) && WriteFile(hand, (char*)utf16_bom+written,
1900                           sizeof(utf16_bom)-written, &tmp, NULL))
1901                   written += tmp;
1902               if (written != sizeof(utf16_bom))
1903               {
1904                   WARN("error writing BOM\n");
1905                   CloseHandle(hand);
1906                   _dosmaperr(GetLastError());
1907                   return *_errno();
1908               }
1909           }
1910       }
1911       else if (access & GENERIC_READ)
1912           oflags = check_bom(hand, oflags, TRUE);
1913   }
1914 
1915   type = GetFileType(hand);
1916   if (type == FILE_TYPE_CHAR)
1917       wxflag |= WX_TTY;
1918   else if (type == FILE_TYPE_PIPE)
1919       wxflag |= WX_PIPE;
1920 
1921   *fd = msvcrt_alloc_fd(hand, wxflag);
1922   if (*fd == -1)
1923       return *_errno();
1924 
1925   if (oflags & _O_WTEXT)
1926       get_ioinfo_nolock(*fd)->exflag |= EF_UTF16|EF_UNK_UNICODE;
1927   else if (oflags & _O_U16TEXT)
1928       get_ioinfo_nolock(*fd)->exflag |= EF_UTF16;
1929   else if (oflags & _O_U8TEXT)
1930       get_ioinfo_nolock(*fd)->exflag |= EF_UTF8;
1931 
1932   TRACE(":fd (%d) handle (%p)\n", *fd, hand);
1933   return 0;
1934 }
1935 
1936 /*********************************************************************
1937  *              _wsopen (MSVCRT.@)
1938  */
1939 int CDECL _wsopen( const wchar_t *path, int oflags, int shflags, ... )
1940 {
1941   int pmode;
1942   int fd;
1943 
1944   if (oflags & _O_CREAT)
1945   {
1946     __ms_va_list ap;
1947 
1948     __ms_va_start(ap, shflags);
1949     pmode = va_arg(ap, int);
1950     __ms_va_end(ap);
1951   }
1952   else
1953     pmode = 0;
1954 
1955   _wsopen_s(&fd, path, oflags, shflags, pmode);
1956   return fd;
1957 }
1958 
1959 /*********************************************************************
1960  *              _sopen_s (MSVCRT.@)
1961  */
1962 int CDECL _sopen_s( int *fd, const char *path, int oflags, int shflags, int pmode )
1963 {
1964     wchar_t *pathW;
1965     int ret;
1966 
1967     if(!MSVCRT_CHECK_PMT(path && (pathW = msvcrt_wstrdupa(path))))
1968         return EINVAL;
1969 
1970     ret = _wsopen_s(fd, pathW, oflags, shflags, pmode);
1971     free(pathW);
1972     return ret;
1973 }
1974 
1975 /*********************************************************************
1976  *              _sopen (MSVCRT.@)
1977  */
1978 int CDECL _sopen( const char *path, int oflags, int shflags, ... )
1979 {
1980   int pmode;
1981   int fd;
1982 
1983   if (oflags & _O_CREAT)
1984   {
1985     va_list ap;
1986 
1987     va_start(ap, shflags);
1988     pmode = va_arg(ap, int);
1989     va_end(ap);
1990   }
1991   else
1992     pmode = 0;
1993 
1994   _sopen_s(&fd, path, oflags, shflags, pmode);
1995   return fd;
1996 }
1997 
1998 /*********************************************************************
1999  *              _open (MSVCRT.@)
2000  */
2001 int CDECL _open( const char *path, int flags, ... )
2002 {
2003   va_list ap;
2004 
2005   if (flags & _O_CREAT)
2006   {
2007     int pmode;
2008     va_start(ap, flags);
2009     pmode = va_arg(ap, int);
2010     va_end(ap);
2011     return _sopen( path, flags, _SH_DENYNO, pmode );
2012   }
2013   else
2014     return _sopen( path, flags, _SH_DENYNO);
2015 }
2016 
2017 /*********************************************************************
2018  *              _wopen (MSVCRT.@)
2019  */
2020 int CDECL _wopen(const wchar_t *path,int flags,...)
2021 {
2022   va_list ap;
2023 
2024   if (flags & _O_CREAT)
2025   {
2026     int pmode;
2027     va_start(ap, flags);
2028     pmode = va_arg(ap, int);
2029     va_end(ap);
2030     return _wsopen( path, flags, _SH_DENYNO, pmode );
2031   }
2032   else
2033     return _wsopen( path, flags, _SH_DENYNO);
2034 }
2035 
2036 /*********************************************************************
2037  *		_creat (MSVCRT.@)
2038  */
2039 int CDECL _creat(const char *path, int flags)
2040 {
2041   int usedFlags = (flags & _O_TEXT)| _O_CREAT| _O_WRONLY| _O_TRUNC;
2042   return _open(path, usedFlags);
2043 }
2044 
2045 /*********************************************************************
2046  *		_wcreat (MSVCRT.@)
2047  */
2048 int CDECL _wcreat(const wchar_t *path, int flags)
2049 {
2050   int usedFlags = (flags & _O_TEXT)| _O_CREAT| _O_WRONLY| _O_TRUNC;
2051   return _wopen(path, usedFlags);
2052 }
2053 
2054 /*********************************************************************
2055  *		_open_osfhandle (MSVCRT.@)
2056  */
2057 int CDECL _open_osfhandle(intptr_t handle, int oflags)
2058 {
2059   DWORD flags;
2060   int fd;
2061 
2062   /* _O_RDONLY (0) always matches, so set the read flag
2063    * MFC's CStdioFile clears O_RDONLY (0)! if it wants to write to the
2064    * file, so set the write flag. It also only sets _O_TEXT if it wants
2065    * text - it never sets _O_BINARY.
2066    */
2067   /* don't let split_oflags() decide the mode if no mode is passed */
2068   if (!(oflags & (_O_BINARY | _O_TEXT)))
2069       oflags |= _O_BINARY;
2070 
2071   flags = GetFileType((HANDLE)handle);
2072   if (flags==FILE_TYPE_UNKNOWN && GetLastError()!=NO_ERROR)
2073   {
2074     _dosmaperr(GetLastError());
2075     return -1;
2076   }
2077 
2078   if (flags == FILE_TYPE_CHAR)
2079     flags = WX_TTY;
2080   else if (flags == FILE_TYPE_PIPE)
2081     flags = WX_PIPE;
2082   else
2083     flags = 0;
2084   flags |= split_oflags(oflags);
2085 
2086   fd = msvcrt_alloc_fd((HANDLE)handle, flags);
2087   TRACE(":handle (%ld) fd (%d) flags 0x%08x\n", handle, fd, flags);
2088   return fd;
2089 }
2090 
2091 /*********************************************************************
2092  *		_rmtmp (MSVCRT.@)
2093  */
2094 int CDECL _rmtmp(void)
2095 {
2096   int num_removed = 0, i;
2097   FILE *file;
2098 
2099   LOCK_FILES();
2100   for (i = 3; i < MSVCRT_stream_idx; i++) {
2101     file = msvcrt_get_file(i);
2102 
2103     if (file->_tmpfname)
2104     {
2105       fclose(file);
2106       num_removed++;
2107     }
2108   }
2109   UNLOCK_FILES();
2110 
2111   if (num_removed)
2112     TRACE(":removed (%d) temp files\n",num_removed);
2113   return num_removed;
2114 }
2115 
2116 static inline int get_utf8_char_len(char ch)
2117 {
2118     if((ch&0xf8) == 0xf0)
2119         return 4;
2120     else if((ch&0xf0) == 0xe0)
2121         return 3;
2122     else if((ch&0xe0) == 0xc0)
2123         return 2;
2124     return 1;
2125 }
2126 
2127 /*********************************************************************
2128  * (internal) read_utf8
2129  */
2130 static int read_utf8(ioinfo *fdinfo, wchar_t *buf, unsigned int count)
2131 {
2132     HANDLE hand = fdinfo->handle;
2133     char min_buf[4], *readbuf, lookahead;
2134     DWORD readbuf_size, pos=0, num_read=1, char_len, i, j;
2135 
2136     /* make the buffer big enough to hold at least one character */
2137     /* read bytes have to fit to output and lookahead buffers */
2138     count /= 2;
2139     readbuf_size = count < 4 ? 4 : count;
2140     if(readbuf_size<=4 || !(readbuf = malloc(readbuf_size))) {
2141         readbuf_size = 4;
2142         readbuf = min_buf;
2143     }
2144 
2145     if(fdinfo->lookahead[0] != '\n') {
2146         readbuf[pos++] = fdinfo->lookahead[0];
2147         fdinfo->lookahead[0] = '\n';
2148 
2149         if(fdinfo->lookahead[1] != '\n') {
2150             readbuf[pos++] = fdinfo->lookahead[1];
2151             fdinfo->lookahead[1] = '\n';
2152 
2153             if(fdinfo->lookahead[2] != '\n') {
2154                 readbuf[pos++] = fdinfo->lookahead[2];
2155                 fdinfo->lookahead[2] = '\n';
2156             }
2157         }
2158     }
2159 
2160     /* NOTE: this case is broken in native dll, reading
2161      *        sometimes fails when small buffer is passed
2162      */
2163     if(count < 4) {
2164         if(!pos && !ReadFile(hand, readbuf, 1, &num_read, NULL)) {
2165             if (GetLastError() == ERROR_BROKEN_PIPE) {
2166                 fdinfo->wxflag |= WX_ATEOF;
2167                 return 0;
2168             }else {
2169                 _dosmaperr(GetLastError());
2170                 return -1;
2171             }
2172         }else if(!num_read) {
2173             fdinfo->wxflag |= WX_ATEOF;
2174             return 0;
2175         }else {
2176             pos++;
2177         }
2178 
2179         char_len = get_utf8_char_len(readbuf[0]);
2180         if(char_len>pos) {
2181             if(ReadFile(hand, readbuf+pos, char_len-pos, &num_read, NULL))
2182                 pos += num_read;
2183         }
2184 
2185         if(readbuf[0] == '\n')
2186             fdinfo->wxflag |= WX_READNL;
2187         else
2188             fdinfo->wxflag &= ~WX_READNL;
2189 
2190         if(readbuf[0] == 0x1a) {
2191             fdinfo->wxflag |= WX_ATEOF;
2192             return 0;
2193         }
2194 
2195         if(readbuf[0] == '\r') {
2196             if(!ReadFile(hand, &lookahead, 1, &num_read, NULL) || num_read!=1)
2197                 buf[0] = '\r';
2198             else if(lookahead == '\n')
2199                 buf[0] = '\n';
2200             else {
2201                 buf[0] = '\r';
2202                 if(fdinfo->wxflag & (WX_PIPE | WX_TTY))
2203                     fdinfo->lookahead[0] = lookahead;
2204                 else
2205                     SetFilePointer(fdinfo->handle, -1, NULL, FILE_CURRENT);
2206             }
2207             return 2;
2208         }
2209 
2210         if(!(num_read = MultiByteToWideChar(CP_UTF8, 0, readbuf, pos, buf, count))) {
2211             _dosmaperr(GetLastError());
2212             return -1;
2213         }
2214 
2215         return num_read*2;
2216     }
2217 
2218     if(!ReadFile(hand, readbuf+pos, readbuf_size-pos, &num_read, NULL)) {
2219         if(pos) {
2220             num_read = 0;
2221         }else if(GetLastError() == ERROR_BROKEN_PIPE) {
2222             fdinfo->wxflag |= WX_ATEOF;
2223             if (readbuf != min_buf) free(readbuf);
2224             return 0;
2225         }else {
2226             _dosmaperr(GetLastError());
2227             if (readbuf != min_buf) free(readbuf);
2228             return -1;
2229         }
2230     }else if(!pos && !num_read) {
2231         fdinfo->wxflag |= WX_ATEOF;
2232         if (readbuf != min_buf) free(readbuf);
2233         return 0;
2234     }
2235 
2236     pos += num_read;
2237     if(readbuf[0] == '\n')
2238         fdinfo->wxflag |= WX_READNL;
2239     else
2240         fdinfo->wxflag &= ~WX_READNL;
2241 
2242     /* Find first byte of last character (may be incomplete) */
2243     for(i=pos-1; i>0 && i>pos-4; i--)
2244         if((readbuf[i]&0xc0) != 0x80)
2245             break;
2246     char_len = get_utf8_char_len(readbuf[i]);
2247     if(char_len+i <= pos)
2248         i += char_len;
2249 
2250     if(fdinfo->wxflag & (WX_PIPE | WX_TTY)) {
2251         if(i < pos)
2252             fdinfo->lookahead[0] = readbuf[i];
2253         if(i+1 < pos)
2254             fdinfo->lookahead[1] = readbuf[i+1];
2255         if(i+2 < pos)
2256             fdinfo->lookahead[2] = readbuf[i+2];
2257     }else if(i < pos) {
2258         SetFilePointer(fdinfo->handle, i-pos, NULL, FILE_CURRENT);
2259     }
2260     pos = i;
2261 
2262     for(i=0, j=0; i<pos; i++) {
2263         if(readbuf[i] == 0x1a) {
2264             fdinfo->wxflag |= WX_ATEOF;
2265             break;
2266         }
2267 
2268         /* strip '\r' if followed by '\n' */
2269         if(readbuf[i] == '\r' && i+1==pos) {
2270             if(fdinfo->lookahead[0] != '\n' || !ReadFile(hand, &lookahead, 1, &num_read, NULL) || !num_read) {
2271                 readbuf[j++] = '\r';
2272             }else if(lookahead == '\n' && j==0) {
2273                 readbuf[j++] = '\n';
2274             }else {
2275                 if(lookahead != '\n')
2276                     readbuf[j++] = '\r';
2277 
2278                 if(fdinfo->wxflag & (WX_PIPE | WX_TTY))
2279                     fdinfo->lookahead[0] = lookahead;
2280                 else
2281                     SetFilePointer(fdinfo->handle, -1, NULL, FILE_CURRENT);
2282             }
2283         }else if(readbuf[i]!='\r' || readbuf[i+1]!='\n') {
2284             readbuf[j++] = readbuf[i];
2285         }
2286     }
2287     pos = j;
2288 
2289     if(!(num_read = MultiByteToWideChar(CP_UTF8, 0, readbuf, pos, buf, count))) {
2290         _dosmaperr(GetLastError());
2291         if (readbuf != min_buf) free(readbuf);
2292         return -1;
2293     }
2294 
2295     if (readbuf != min_buf) free(readbuf);
2296     return num_read*2;
2297 }
2298 
2299 /*********************************************************************
2300  * (internal) read_i
2301  *
2302  * When reading \r as last character in text mode, read() positions
2303  * the file pointer on the \r character while getc() goes on to
2304  * the following \n
2305  */
2306 static int read_i(int fd, ioinfo *fdinfo, void *buf, unsigned int count)
2307 {
2308     DWORD num_read, utf16;
2309     char *bufstart = buf;
2310 
2311     if (count == 0)
2312         return 0;
2313 
2314     if (fdinfo->wxflag & WX_ATEOF) {
2315         TRACE("already at EOF, returning 0\n");
2316         return 0;
2317     }
2318     /* Don't trace small reads, it gets *very* annoying */
2319     if (count > 4)
2320         TRACE(":fd (%d) handle (%p) buf (%p) len (%d)\n", fd, fdinfo->handle, buf, count);
2321     if (fdinfo->handle == INVALID_HANDLE_VALUE)
2322     {
2323         *_errno() = EBADF;
2324         return -1;
2325     }
2326 
2327     utf16 = (fdinfo->exflag & EF_UTF16) != 0;
2328     if (((fdinfo->exflag&EF_UTF8) || utf16) && count&1)
2329     {
2330         *_errno() = EINVAL;
2331         return -1;
2332     }
2333 
2334     if((fdinfo->wxflag&WX_TEXT) && (fdinfo->exflag&EF_UTF8))
2335         return read_utf8(fdinfo, buf, count);
2336 
2337     if (fdinfo->lookahead[0]!='\n' || ReadFile(fdinfo->handle, bufstart, count, &num_read, NULL))
2338     {
2339         if (fdinfo->lookahead[0] != '\n')
2340         {
2341             bufstart[0] = fdinfo->lookahead[0];
2342             fdinfo->lookahead[0] = '\n';
2343 
2344             if (utf16)
2345             {
2346                 bufstart[1] =  fdinfo->lookahead[1];
2347                 fdinfo->lookahead[1] = '\n';
2348             }
2349 
2350             if(count>1+utf16 && ReadFile(fdinfo->handle, bufstart+1+utf16, count-1-utf16, &num_read, NULL))
2351                 num_read += 1+utf16;
2352             else
2353                 num_read = 1+utf16;
2354         }
2355 
2356         if(utf16 && (num_read&1))
2357         {
2358             /* msvcr90 uses uninitialized value from the buffer in this case */
2359             /* msvcrt ignores additional data */
2360             ERR("got odd number of bytes in UTF16 mode\n");
2361             num_read--;
2362         }
2363 
2364         if (count != 0 && num_read == 0)
2365         {
2366             fdinfo->wxflag |= WX_ATEOF;
2367             TRACE(":EOF %s\n",debugstr_an(buf,num_read));
2368         }
2369         else if (fdinfo->wxflag & WX_TEXT)
2370         {
2371             DWORD i, j;
2372 
2373             if (bufstart[0]=='\n' && (!utf16 || bufstart[1]==0))
2374                 fdinfo->wxflag |= WX_READNL;
2375             else
2376                 fdinfo->wxflag &= ~WX_READNL;
2377 
2378             for (i=0, j=0; i<num_read; i+=1+utf16)
2379             {
2380                 /* in text mode, a ctrl-z signals EOF */
2381                 if (bufstart[i]==0x1a && (!utf16 || bufstart[i+1]==0))
2382                 {
2383                     fdinfo->wxflag |= WX_ATEOF;
2384                     TRACE(":^Z EOF %s\n",debugstr_an(buf,num_read));
2385                     break;
2386                 }
2387 
2388                 /* in text mode, strip \r if followed by \n */
2389                 if (bufstart[i]=='\r' && (!utf16 || bufstart[i+1]==0) && i+1+utf16==num_read)
2390                 {
2391                     char lookahead[2];
2392                     DWORD len;
2393 
2394                     lookahead[1] = '\n';
2395                     if (ReadFile(fdinfo->handle, lookahead, 1+utf16, &len, NULL) && len)
2396                     {
2397                         if(lookahead[0]=='\n' && (!utf16 || lookahead[1]==0) && j==0)
2398                         {
2399                             bufstart[j++] = '\n';
2400                             if(utf16) bufstart[j++] = 0;
2401                         }
2402                         else
2403                         {
2404                             if(lookahead[0]!='\n' || (utf16 && lookahead[1]!=0))
2405                             {
2406                                 bufstart[j++] = '\r';
2407                                 if(utf16) bufstart[j++] = 0;
2408                             }
2409 
2410                             if (fdinfo->wxflag & (WX_PIPE | WX_TTY))
2411                             {
2412                                 if (lookahead[0]=='\n' && (!utf16 || !lookahead[1]))
2413                                 {
2414                                     bufstart[j++] = '\n';
2415                                     if (utf16) bufstart[j++] = 0;
2416                                 }
2417                                 else
2418                                 {
2419                                     fdinfo->lookahead[0] = lookahead[0];
2420                                     fdinfo->lookahead[1] = lookahead[1];
2421                                 }
2422                             }
2423                             else
2424                                 SetFilePointer(fdinfo->handle, -1-utf16, NULL, FILE_CURRENT);
2425                         }
2426                     }
2427                     else
2428                     {
2429                         bufstart[j++] = '\r';
2430                         if(utf16) bufstart[j++] = 0;
2431                     }
2432                 }
2433                 else if((bufstart[i]!='\r' || (utf16 && bufstart[i+1]!=0))
2434                         || (bufstart[i+1+utf16]!='\n' || (utf16 && bufstart[i+3]!=0)))
2435                 {
2436                     bufstart[j++] = bufstart[i];
2437                     if(utf16) bufstart[j++] = bufstart[i+1];
2438                 }
2439             }
2440             num_read = j;
2441         }
2442     }
2443     else
2444     {
2445         if (GetLastError() == ERROR_BROKEN_PIPE)
2446         {
2447             TRACE(":end-of-pipe\n");
2448             fdinfo->wxflag |= WX_ATEOF;
2449             return 0;
2450         }
2451         else
2452         {
2453             TRACE(":failed-last error (%d)\n",GetLastError());
2454             return -1;
2455         }
2456     }
2457 
2458     if (count > 4)
2459         TRACE("(%u), %s\n",num_read,debugstr_an(buf, num_read));
2460     return num_read;
2461 }
2462 
2463 /*********************************************************************
2464  *		_read (MSVCRT.@)
2465  */
2466 int CDECL _read(int fd, void *buf, unsigned int count)
2467 {
2468     ioinfo *info = get_ioinfo(fd);
2469     int num_read = read_i(fd, info, buf, count);
2470     release_ioinfo(info);
2471     return num_read;
2472 }
2473 
2474 /*********************************************************************
2475  *		_setmode (MSVCRT.@)
2476  */
2477 int CDECL _setmode(int fd,int mode)
2478 {
2479     ioinfo *info = get_ioinfo(fd);
2480     int ret = info->wxflag & WX_TEXT ? _O_TEXT : _O_BINARY;
2481     if(ret==_O_TEXT && (info->exflag & (EF_UTF8|EF_UTF16)))
2482         ret = _O_WTEXT;
2483 
2484     if(mode!=_O_TEXT && mode!=_O_BINARY && mode!=_O_WTEXT
2485                 && mode!=_O_U16TEXT && mode!=_O_U8TEXT) {
2486         *_errno() = EINVAL;
2487         release_ioinfo(info);
2488         return -1;
2489     }
2490 
2491     if(info == &__badioinfo) {
2492         *_errno() = EBADF;
2493         return EOF;
2494     }
2495 
2496     if(mode == _O_BINARY) {
2497         info->wxflag &= ~WX_TEXT;
2498         info->exflag &= ~(EF_UTF8|EF_UTF16);
2499         release_ioinfo(info);
2500         return ret;
2501     }
2502 
2503     info->wxflag |= WX_TEXT;
2504     if(mode == _O_TEXT)
2505         info->exflag &= ~(EF_UTF8|EF_UTF16);
2506     else if(mode == _O_U8TEXT)
2507         info->exflag = (info->exflag & ~EF_UTF16) | EF_UTF8;
2508     else
2509         info->exflag = (info->exflag & ~EF_UTF8) | EF_UTF16;
2510 
2511     release_ioinfo(info);
2512     return ret;
2513 
2514 }
2515 
2516 /*********************************************************************
2517  *		_tell (MSVCRT.@)
2518  */
2519 long CDECL _tell(int fd)
2520 {
2521   return _lseek(fd, 0, SEEK_CUR);
2522 }
2523 
2524 /*********************************************************************
2525  *		_telli64 (MSVCRT.@)
2526  */
2527 __int64 CDECL _telli64(int fd)
2528 {
2529   return _lseeki64(fd, 0, SEEK_CUR);
2530 }
2531 
2532 /*********************************************************************
2533  *		_tempnam (MSVCRT.@)
2534  */
2535 char * CDECL _tempnam(const char *dir, const char *prefix)
2536 {
2537   char tmpbuf[MAX_PATH];
2538   const char *tmp_dir = getenv("TMP");
2539 
2540   if (tmp_dir) dir = tmp_dir;
2541 
2542   TRACE("dir (%s) prefix (%s)\n",dir,prefix);
2543   if (GetTempFileNameA(dir,prefix,0,tmpbuf))
2544   {
2545     TRACE("got name (%s)\n",tmpbuf);
2546     DeleteFileA(tmpbuf);
2547     return _strdup(tmpbuf);
2548   }
2549   TRACE("failed (%d)\n",GetLastError());
2550   return NULL;
2551 }
2552 
2553 /*********************************************************************
2554  *		_wtempnam (MSVCRT.@)
2555  */
2556 wchar_t * CDECL _wtempnam(const wchar_t *dir, const wchar_t *prefix)
2557 {
2558   wchar_t tmpbuf[MAX_PATH];
2559 
2560   TRACE("dir (%s) prefix (%s)\n",debugstr_w(dir),debugstr_w(prefix));
2561   if (GetTempFileNameW(dir,prefix,0,tmpbuf))
2562   {
2563     TRACE("got name (%s)\n",debugstr_w(tmpbuf));
2564     DeleteFileW(tmpbuf);
2565     return _wcsdup(tmpbuf);
2566   }
2567   TRACE("failed (%d)\n",GetLastError());
2568   return NULL;
2569 }
2570 
2571 /*********************************************************************
2572  *		_umask (MSVCRT.@)
2573  */
2574 int CDECL _umask(int umask)
2575 {
2576   int old_umask = MSVCRT_umask;
2577   TRACE("(%d)\n",umask);
2578   MSVCRT_umask = umask;
2579   return old_umask;
2580 }
2581 
2582 /*********************************************************************
2583  *		_write (MSVCRT.@)
2584  */
2585 int CDECL _write(int fd, const void* buf, unsigned int count)
2586 {
2587     DWORD num_written;
2588     ioinfo *info = get_ioinfo(fd);
2589     HANDLE hand = info->handle;
2590 
2591     /* Don't trace small writes, it gets *very* annoying */
2592 #if 0
2593     if (count > 32)
2594         TRACE(":fd (%d) handle (%d) buf (%p) len (%d)\n",fd,hand,buf,count);
2595 #endif
2596     if (hand == INVALID_HANDLE_VALUE)
2597     {
2598         *_errno() = EBADF;
2599         release_ioinfo(info);
2600         return -1;
2601     }
2602 
2603     if (((info->exflag&EF_UTF8) || (info->exflag&EF_UTF16)) && count&1)
2604     {
2605         *_errno() = EINVAL;
2606         release_ioinfo(info);
2607         return -1;
2608     }
2609 
2610     /* If appending, go to EOF */
2611     if (info->wxflag & WX_APPEND)
2612         _lseek(fd, 0, FILE_END);
2613 
2614     if (!(info->wxflag & WX_TEXT))
2615     {
2616         if (WriteFile(hand, buf, count, &num_written, NULL)
2617                 &&  (num_written == count))
2618         {
2619             release_ioinfo(info);
2620             return num_written;
2621         }
2622         TRACE("WriteFile (fd %d, hand %p) failed-last error (%d)\n", fd,
2623                 hand, GetLastError());
2624         *_errno() = ENOSPC;
2625     }
2626     else
2627     {
2628         unsigned int i, j, nr_lf, size;
2629         char *p = NULL;
2630         const char *q;
2631         const char *s = buf, *buf_start = buf;
2632 
2633         if (!(info->exflag & (EF_UTF8|EF_UTF16)))
2634         {
2635             /* find number of \n */
2636             for (nr_lf=0, i=0; i<count; i++)
2637                 if (s[i] == '\n')
2638                     nr_lf++;
2639             if (nr_lf)
2640             {
2641                 size = count+nr_lf;
2642                 if ((q = p = malloc(size)))
2643                 {
2644                     for (s = buf, i = 0, j = 0; i < count; i++)
2645                     {
2646                         if (s[i] == '\n')
2647                             p[j++] = '\r';
2648                         p[j++] = s[i];
2649                     }
2650                 }
2651                 else
2652                 {
2653                     FIXME("Malloc failed\n");
2654                     nr_lf = 0;
2655                     size = count;
2656                     q = buf;
2657                 }
2658             }
2659             else
2660             {
2661                 size = count;
2662                 q = buf;
2663             }
2664         }
2665         else if (info->exflag & EF_UTF16)
2666         {
2667             for (nr_lf=0, i=0; i<count; i+=2)
2668                 if (s[i]=='\n' && s[i+1]==0)
2669                     nr_lf += 2;
2670             if (nr_lf)
2671             {
2672                 size = count+nr_lf;
2673                 if ((q = p = malloc(size)))
2674                 {
2675                     for (s=buf, i=0, j=0; i<count; i++)
2676                     {
2677                         if (s[i]=='\n' && s[i+1]==0)
2678                         {
2679                             p[j++] = '\r';
2680                             p[j++] = 0;
2681                         }
2682                         p[j++] = s[i++];
2683                         p[j++] = s[i];
2684                     }
2685                 }
2686                 else
2687                 {
2688                     FIXME("Malloc failed\n");
2689                     nr_lf = 0;
2690                     size = count;
2691                     q = buf;
2692                 }
2693             }
2694             else
2695             {
2696                 size = count;
2697                 q = buf;
2698             }
2699         }
2700         else
2701         {
2702             DWORD conv_len;
2703 
2704             for(nr_lf=0, i=0; i<count; i+=2)
2705                 if (s[i]=='\n' && s[i+1]==0)
2706                     nr_lf++;
2707 
2708             conv_len = WideCharToMultiByte(CP_UTF8, 0, (WCHAR*)buf, count/2, NULL, 0, NULL, NULL);
2709             if(!conv_len) {
2710                 _dosmaperr(GetLastError());
2711                 free(p);
2712                 release_ioinfo(info);
2713                 return -1;
2714             }
2715 
2716             size = conv_len+nr_lf;
2717             if((p = malloc(count+nr_lf*2+size)))
2718             {
2719                 for (s=buf, i=0, j=0; i<count; i++)
2720                 {
2721                     if (s[i]=='\n' && s[i+1]==0)
2722                     {
2723                         p[j++] = '\r';
2724                         p[j++] = 0;
2725                     }
2726                     p[j++] = s[i++];
2727                     p[j++] = s[i];
2728                 }
2729                 q = p+count+nr_lf*2;
2730                 WideCharToMultiByte(CP_UTF8, 0, (WCHAR*)p, count/2+nr_lf,
2731                         p+count+nr_lf*2, conv_len+nr_lf, NULL, NULL);
2732             }
2733             else
2734             {
2735                 FIXME("Malloc failed\n");
2736                 nr_lf = 0;
2737                 size = count;
2738                 q = buf;
2739             }
2740         }
2741 
2742         if (!WriteFile(hand, q, size, &num_written, NULL))
2743             num_written = -1;
2744         release_ioinfo(info);
2745         if(p)
2746             free(p);
2747         if (num_written != size)
2748         {
2749             TRACE("WriteFile (fd %d, hand %p) failed-last error (%d), num_written %d\n",
2750                     fd, hand, GetLastError(), num_written);
2751             *_errno() = ENOSPC;
2752             return s - buf_start;
2753         }
2754         return count;
2755     }
2756 
2757     release_ioinfo(info);
2758     return -1;
2759 }
2760 
2761 /*********************************************************************
2762  *		_putw (MSVCRT.@)
2763  */
2764 int CDECL _putw(int val, FILE* file)
2765 {
2766   int len;
2767 
2768   _lock_file(file);
2769   len = _write(file->_file, &val, sizeof(val));
2770   if (len == sizeof(val)) {
2771     _unlock_file(file);
2772     return val;
2773   }
2774 
2775   file->_flag |= _IOERR;
2776   _unlock_file(file);
2777   return EOF;
2778 }
2779 
2780 /*********************************************************************
2781  *		fclose (MSVCRT.@)
2782  */
2783 int CDECL fclose(FILE* file)
2784 {
2785   int r, flag;
2786 
2787   _lock_file(file);
2788   flag = file->_flag;
2789   free(file->_tmpfname);
2790   file->_tmpfname = NULL;
2791   /* flush stdio buffers */
2792   if(file->_flag & _IOWRT)
2793       fflush(file);
2794   if(file->_flag & _IOMYBUF)
2795       free(file->_base);
2796 
2797   r=_close(file->_file);
2798 
2799   file->_flag = 0;
2800   _unlock_file(file);
2801   if(file<_iob || file>=_iob+_IOB_ENTRIES)
2802       DeleteCriticalSection(&((file_crit*)file)->crit);
2803 
2804   if(file == msvcrt_get_file(MSVCRT_stream_idx-1)) {
2805     while(MSVCRT_stream_idx>3 && !file->_flag) {
2806       MSVCRT_stream_idx--;
2807       file = msvcrt_get_file(MSVCRT_stream_idx-1);
2808     }
2809   }
2810 
2811   return ((r == -1) || (flag & _IOERR) ? EOF : 0);
2812 }
2813 
2814 /*********************************************************************
2815  *		feof (MSVCRT.@)
2816  */
2817 int CDECL feof(FILE* file)
2818 {
2819     return file->_flag & _IOEOF;
2820 }
2821 
2822 /*********************************************************************
2823  *		ferror (MSVCRT.@)
2824  */
2825 int CDECL ferror(FILE* file)
2826 {
2827     return file->_flag & _IOERR;
2828 }
2829 
2830 /*********************************************************************
2831  *		_filbuf (MSVCRT.@)
2832  */
2833 int CDECL _filbuf(FILE* file)
2834 {
2835     unsigned char c;
2836     _lock_file(file);
2837 
2838     if(file->_flag & _IOSTRG) {
2839         _unlock_file(file);
2840         return EOF;
2841     }
2842 
2843     /* Allocate buffer if needed */
2844     if(!(file->_flag & (_IONBF | _IOMYBUF | _USERBUF)))
2845         msvcrt_alloc_buffer(file);
2846 
2847     if(!(file->_flag & _IOREAD)) {
2848         if(file->_flag & _IORW)
2849             file->_flag |= _IOREAD;
2850         else {
2851             _unlock_file(file);
2852             return EOF;
2853         }
2854     }
2855 
2856     if(!(file->_flag & (_IOMYBUF | _USERBUF))) {
2857         int r;
2858         if ((r = _read(file->_file,&c,1)) != 1) {
2859             file->_flag |= (r == 0) ? _IOEOF : _IOERR;
2860             _unlock_file(file);
2861             return EOF;
2862         }
2863 
2864         _unlock_file(file);
2865         return c;
2866     } else {
2867         file->_cnt = _read(file->_file, file->_base, file->_bufsiz);
2868         if(file->_cnt<=0) {
2869             file->_flag |= (file->_cnt == 0) ? _IOEOF : _IOERR;
2870             file->_cnt = 0;
2871             _unlock_file(file);
2872             return EOF;
2873         }
2874 
2875         file->_cnt--;
2876         file->_ptr = file->_base+1;
2877         c = *(unsigned char *)file->_base;
2878         _unlock_file(file);
2879         return c;
2880     }
2881 }
2882 
2883 /*********************************************************************
2884  *		fgetc (MSVCRT.@)
2885  */
2886 int CDECL fgetc(FILE* file)
2887 {
2888   unsigned char *i;
2889   unsigned int j;
2890 
2891   _lock_file(file);
2892   if (file->_cnt>0) {
2893     file->_cnt--;
2894     i = (unsigned char *)file->_ptr++;
2895     j = *i;
2896   } else
2897     j = _filbuf(file);
2898 
2899   _unlock_file(file);
2900   return j;
2901 }
2902 
2903 /*********************************************************************
2904  *		_fgetchar (MSVCRT.@)
2905  */
2906 int CDECL _fgetchar(void)
2907 {
2908   return fgetc(stdin);
2909 }
2910 
2911 /*********************************************************************
2912  *		fgets (MSVCRT.@)
2913  */
2914 char * CDECL fgets(char *s, int size, FILE* file)
2915 {
2916   int    cc = EOF;
2917   char * buf_start = s;
2918 
2919   TRACE(":file(%p) fd (%d) str (%p) len (%d)\n",
2920 	file,file->_file,s,size);
2921 
2922   _lock_file(file);
2923 
2924   while ((size >1) && (cc = fgetc(file)) != EOF && cc != '\n')
2925     {
2926       *s++ = (char)cc;
2927       size --;
2928     }
2929   if ((cc == EOF) && (s == buf_start)) /* If nothing read, return 0*/
2930   {
2931     TRACE(":nothing read\n");
2932     _unlock_file(file);
2933     return NULL;
2934   }
2935   if ((cc != EOF) && (size > 1))
2936     *s++ = cc;
2937   *s = '\0';
2938   TRACE(":got %s\n", debugstr_a(buf_start));
2939   _unlock_file(file);
2940   return buf_start;
2941 }
2942 
2943 /*********************************************************************
2944  *		fgetwc (MSVCRT.@)
2945  */
2946 wint_t CDECL fgetwc(FILE* file)
2947 {
2948     wint_t ret;
2949     int ch;
2950 
2951     _lock_file(file);
2952 
2953     if((get_ioinfo_nolock(file->_file)->exflag & (EF_UTF8 | EF_UTF16))
2954             || !(get_ioinfo_nolock(file->_file)->wxflag & WX_TEXT)) {
2955         char *p;
2956 
2957         for(p=(char*)&ret; (wint_t*)p<&ret+1; p++) {
2958             ch = fgetc(file);
2959             if(ch == EOF) {
2960                 ret = WEOF;
2961                 break;
2962             }
2963             *p = (char)ch;
2964         }
2965     }else {
2966         char mbs[MB_LEN_MAX];
2967         int len = 0;
2968 
2969         ch = fgetc(file);
2970         if(ch != EOF) {
2971             mbs[0] = (char)ch;
2972             if(isleadbyte((unsigned char)mbs[0])) {
2973                 ch = fgetc(file);
2974                 if(ch != EOF) {
2975                     mbs[1] = (char)ch;
2976                     len = 2;
2977                 }
2978             }else {
2979                 len = 1;
2980             }
2981         }
2982 
2983         if(!len || mbtowc(&ret, mbs, len)==-1)
2984             ret = WEOF;
2985     }
2986 
2987     _unlock_file(file);
2988     return ret;
2989 }
2990 
2991 /*********************************************************************
2992  *		_getw (MSVCRT.@)
2993  */
2994 int CDECL _getw(FILE* file)
2995 {
2996   char *ch;
2997   int i, k;
2998   unsigned int j;
2999   ch = (char *)&i;
3000 
3001   _lock_file(file);
3002   for (j=0; j<sizeof(int); j++) {
3003     k = fgetc(file);
3004     if (k == EOF) {
3005       file->_flag |= _IOEOF;
3006       _unlock_file(file);
3007       return EOF;
3008     }
3009     ch[j] = k;
3010   }
3011 
3012   _unlock_file(file);
3013   return i;
3014 }
3015 
3016 /*********************************************************************
3017  *		getwc (MSVCRT.@)
3018  */
3019 wint_t CDECL getwc(FILE* file)
3020 {
3021   return fgetwc(file);
3022 }
3023 
3024 /*********************************************************************
3025  *		_fgetwchar (MSVCRT.@)
3026  */
3027 wint_t CDECL _fgetwchar(void)
3028 {
3029   return fgetwc(stdin);
3030 }
3031 
3032 /*********************************************************************
3033  *		getwchar (MSVCRT.@)
3034  */
3035 wint_t CDECL getwchar(void)
3036 {
3037   return _fgetwchar();
3038 }
3039 
3040 /*********************************************************************
3041  *              fgetws (MSVCRT.@)
3042  */
3043 wchar_t * CDECL fgetws(wchar_t *s, int size, FILE* file)
3044 {
3045   int    cc = WEOF;
3046   wchar_t * buf_start = s;
3047 
3048   TRACE(":file(%p) fd (%d) str (%p) len (%d)\n",
3049         file,file->_file,s,size);
3050 
3051   _lock_file(file);
3052 
3053   while ((size >1) && (cc = fgetwc(file)) != WEOF && cc != '\n')
3054     {
3055       *s++ = (char)cc;
3056       size --;
3057     }
3058   if ((cc == WEOF) && (s == buf_start)) /* If nothing read, return 0*/
3059   {
3060     TRACE(":nothing read\n");
3061     _unlock_file(file);
3062     return NULL;
3063   }
3064   if ((cc != WEOF) && (size > 1))
3065     *s++ = cc;
3066   *s = 0;
3067   TRACE(":got %s\n", debugstr_w(buf_start));
3068   _unlock_file(file);
3069   return buf_start;
3070 }
3071 
3072 /*********************************************************************
3073  *		fwrite (MSVCRT.@)
3074  */
3075 size_t CDECL fwrite(const void *ptr, size_t size, size_t nmemb, FILE* file)
3076 {
3077     size_t wrcnt=size * nmemb;
3078     int written = 0;
3079     if (size == 0)
3080         return 0;
3081 
3082     _lock_file(file);
3083 
3084     while(wrcnt) {
3085 #ifndef __REACTOS__
3086         if(file->_cnt < 0) {
3087             WARN("negative file->_cnt value in %p\n", file);
3088             file->_flag |= MSVCRT__IOERR;
3089             break;
3090         } else
3091 #endif
3092         if(file->_cnt) {
3093             int pcnt=(file->_cnt>wrcnt)? wrcnt: file->_cnt;
3094             memcpy(file->_ptr, ptr, pcnt);
3095             file->_cnt -= pcnt;
3096             file->_ptr += pcnt;
3097             written += pcnt;
3098             wrcnt -= pcnt;
3099             ptr = (const char*)ptr + pcnt;
3100         } else if((file->_flag & _IONBF)
3101                 || ((file->_flag & (_IOMYBUF | _USERBUF)) && wrcnt >= file->_bufsiz)
3102                 || (!(file->_flag & (_IOMYBUF | _USERBUF)) && wrcnt >= MSVCRT_INTERNAL_BUFSIZ)) {
3103             size_t pcnt;
3104             int bufsiz;
3105 
3106             if(file->_flag & _IONBF)
3107                 bufsiz = 1;
3108             else if(!(file->_flag & (_IOMYBUF | _USERBUF)))
3109                 bufsiz = MSVCRT_INTERNAL_BUFSIZ;
3110             else
3111                 bufsiz = file->_bufsiz;
3112 
3113             pcnt = (wrcnt / bufsiz) * bufsiz;
3114 
3115             if(msvcrt_flush_buffer(file) == EOF)
3116                 break;
3117 
3118             if(_write(file->_file, ptr, pcnt) <= 0) {
3119                 file->_flag |= _IOERR;
3120                 break;
3121             }
3122             written += pcnt;
3123             wrcnt -= pcnt;
3124             ptr = (const char*)ptr + pcnt;
3125         } else {
3126             if(_flsbuf(*(const char*)ptr, file) == EOF)
3127                 break;
3128             written++;
3129             wrcnt--;
3130             ptr = (const char*)ptr + 1;
3131         }
3132     }
3133 
3134     _unlock_file(file);
3135     return written / size;
3136 }
3137 
3138 /*********************************************************************
3139  *		fputwc (MSVCRT.@)
3140  * FORKED for ReactOS, don't sync with Wine!
3141  * References:
3142  *   - http://jira.reactos.org/browse/CORE-6495
3143  *   - http://bugs.winehq.org/show_bug.cgi?id=8598
3144  */
3145 wint_t CDECL fputwc(wchar_t c, FILE* stream)
3146 {
3147     /* If this is a real file stream (and not some temporary one for
3148        sprintf-like functions), check whether it is opened in text mode.
3149        In this case, we have to perform an implicit conversion to ANSI. */
3150     if (!(stream->_flag & _IOSTRG) && get_ioinfo_nolock(stream->_file)->wxflag & WX_TEXT)
3151     {
3152         /* Convert to multibyte in text mode */
3153         char mbc[MB_LEN_MAX];
3154         int mb_return;
3155 
3156         mb_return = wctomb(mbc, c);
3157 
3158         if(mb_return == -1)
3159             return WEOF;
3160 
3161         /* Output all characters */
3162         if (fwrite(mbc, mb_return, 1, stream) != 1)
3163             return WEOF;
3164     }
3165     else
3166     {
3167         if (fwrite(&c, sizeof(c), 1, stream) != 1)
3168             return WEOF;
3169     }
3170 
3171     return c;
3172 }
3173 
3174 /*********************************************************************
3175  *		_fputwchar (MSVCRT.@)
3176  */
3177 wint_t CDECL _fputwchar(wint_t wc)
3178 {
3179   return fputwc(wc, stdout);
3180 }
3181 
3182 /*********************************************************************
3183  *		_wfsopen (MSVCRT.@)
3184  */
3185 FILE * CDECL _wfsopen(const wchar_t *path, const wchar_t *mode, int share)
3186 {
3187   FILE* file;
3188   int open_flags, stream_flags, fd;
3189 
3190   TRACE("(%s,%s)\n", debugstr_w(path), debugstr_w(mode));
3191 
3192   /* map mode string to open() flags. "man fopen" for possibilities. */
3193   if (msvcrt_get_flags(mode, &open_flags, &stream_flags) == -1)
3194       return NULL;
3195 
3196   LOCK_FILES();
3197   fd = _wsopen(path, open_flags, share, _S_IREAD | _S_IWRITE);
3198   if (fd < 0)
3199     file = NULL;
3200   else if ((file = msvcrt_alloc_fp()) && msvcrt_init_fp(file, fd, stream_flags)
3201    != -1)
3202     TRACE(":fd (%d) mode (%s) FILE* (%p)\n", fd, debugstr_w(mode), file);
3203   else if (file)
3204   {
3205     file->_flag = 0;
3206     file = NULL;
3207   }
3208 
3209   TRACE(":got (%p)\n",file);
3210   if (fd >= 0 && !file)
3211     _close(fd);
3212   UNLOCK_FILES();
3213   return file;
3214 }
3215 
3216 /*********************************************************************
3217  *		_fsopen (MSVCRT.@)
3218  */
3219 FILE * CDECL _fsopen(const char *path, const char *mode, int share)
3220 {
3221     FILE *ret;
3222     wchar_t *pathW = NULL, *modeW = NULL;
3223 
3224     if (path && !(pathW = msvcrt_wstrdupa(path))) {
3225         _invalid_parameter(NULL, NULL, NULL, 0, 0);
3226         *_errno() = EINVAL;
3227         return NULL;
3228     }
3229     if (mode && !(modeW = msvcrt_wstrdupa(mode)))
3230     {
3231         free(pathW);
3232         _invalid_parameter(NULL, NULL, NULL, 0, 0);
3233         *_errno() = EINVAL;
3234         return NULL;
3235     }
3236 
3237     ret = _wfsopen(pathW, modeW, share);
3238 
3239     free(pathW);
3240     free(modeW);
3241     return ret;
3242 }
3243 
3244 /*********************************************************************
3245  *		fopen (MSVCRT.@)
3246  */
3247 FILE * CDECL fopen(const char *path, const char *mode)
3248 {
3249     return _fsopen( path, mode, _SH_DENYNO );
3250 }
3251 
3252 /*********************************************************************
3253  *              fopen_s (MSVCRT.@)
3254  */
3255 int CDECL fopen_s(FILE** pFile,
3256         const char *filename, const char *mode)
3257 {
3258     if (!MSVCRT_CHECK_PMT(pFile != NULL)) return EINVAL;
3259     if (!MSVCRT_CHECK_PMT(filename != NULL)) return EINVAL;
3260     if (!MSVCRT_CHECK_PMT(mode != NULL)) return EINVAL;
3261 
3262     *pFile = fopen(filename, mode);
3263 
3264     if(!*pFile)
3265         return *_errno();
3266     return 0;
3267 }
3268 
3269 /*********************************************************************
3270  *		_wfopen (MSVCRT.@)
3271  */
3272 FILE * CDECL _wfopen(const wchar_t *path, const wchar_t *mode)
3273 {
3274     return _wfsopen( path, mode, _SH_DENYNO );
3275 }
3276 
3277 /*********************************************************************
3278  *		_wfopen_s (MSVCRT.@)
3279  */
3280 int CDECL _wfopen_s(FILE** pFile, const wchar_t *filename,
3281         const wchar_t *mode)
3282 {
3283     if (!MSVCRT_CHECK_PMT(pFile != NULL) || !MSVCRT_CHECK_PMT(filename != NULL) ||
3284         !MSVCRT_CHECK_PMT(mode != NULL)) {
3285         *_errno() = EINVAL;
3286         return EINVAL;
3287     }
3288 
3289     *pFile = _wfopen(filename, mode);
3290 
3291     if(!*pFile)
3292         return *_errno();
3293     return 0;
3294 }
3295 
3296 /* fputc calls _flsbuf which calls fputc */
3297 int CDECL _flsbuf(int c, FILE* file);
3298 
3299 /*********************************************************************
3300  *		fputc (MSVCRT.@)
3301  */
3302 int CDECL fputc(int c, FILE* file)
3303 {
3304   int res;
3305 
3306   _lock_file(file);
3307   if(file->_cnt>0) {
3308     *file->_ptr++=c;
3309     file->_cnt--;
3310     if (c == '\n')
3311     {
3312       res = msvcrt_flush_buffer(file);
3313       _unlock_file(file);
3314       return res ? res : c;
3315     }
3316     else {
3317       _unlock_file(file);
3318       return c & 0xff;
3319     }
3320   } else {
3321     res = _flsbuf(c, file);
3322     _unlock_file(file);
3323     return res;
3324   }
3325 }
3326 
3327 /*********************************************************************
3328  *		_fputchar (MSVCRT.@)
3329  */
3330 int CDECL _fputchar(int c)
3331 {
3332   return fputc(c, stdout);
3333 }
3334 
3335 /*********************************************************************
3336  *		fread (MSVCRT.@)
3337  */
3338 size_t CDECL fread(void *ptr, size_t size, size_t nmemb, FILE* file)
3339 {
3340   size_t rcnt=size * nmemb;
3341   size_t read=0;
3342   size_t pread=0;
3343 
3344   if(!rcnt)
3345 	return 0;
3346 
3347   _lock_file(file);
3348 
3349   /* first buffered data */
3350   if(file->_cnt>0) {
3351 	int pcnt= (rcnt>file->_cnt)? file->_cnt:rcnt;
3352 	memcpy(ptr, file->_ptr, pcnt);
3353 	file->_cnt -= pcnt;
3354 	file->_ptr += pcnt;
3355 	read += pcnt ;
3356 	rcnt -= pcnt ;
3357         ptr = (char*)ptr + pcnt;
3358   } else if(!(file->_flag & _IOREAD )) {
3359 	if(file->_flag & _IORW) {
3360 		file->_flag |= _IOREAD;
3361 	} else {
3362         _unlock_file(file);
3363         return 0;
3364     }
3365   }
3366 
3367   if(rcnt>0 && !(file->_flag & (_IONBF | _IOMYBUF | _USERBUF)))
3368       msvcrt_alloc_buffer(file);
3369 
3370   while(rcnt>0)
3371   {
3372     int i;
3373     if (!file->_cnt && rcnt<BUFSIZ && (file->_flag & (_IOMYBUF | _USERBUF))) {
3374       file->_cnt = _read(file->_file, file->_base, file->_bufsiz);
3375       file->_ptr = file->_base;
3376       i = (file->_cnt<rcnt) ? file->_cnt : rcnt;
3377       /* If the buffer fill reaches eof but fread wouldn't, clear eof. */
3378       if (i > 0 && i < file->_cnt) {
3379         get_ioinfo_nolock(file->_file)->wxflag &= ~WX_ATEOF;
3380         file->_flag &= ~_IOEOF;
3381       }
3382       if (i > 0) {
3383         memcpy(ptr, file->_ptr, i);
3384         file->_cnt -= i;
3385         file->_ptr += i;
3386       }
3387     } else if (rcnt > INT_MAX) {
3388       i = _read(file->_file, ptr, INT_MAX);
3389     } else if (rcnt < BUFSIZ) {
3390       i = _read(file->_file, ptr, rcnt);
3391     } else {
3392       i = _read(file->_file, ptr, rcnt - BUFSIZ/2);
3393     }
3394     pread += i;
3395     rcnt -= i;
3396     ptr = (char *)ptr+i;
3397     /* expose feof condition in the flags
3398      * MFC tests file->_flag for feof, and doesn't call feof())
3399      */
3400     if (get_ioinfo_nolock(file->_file)->wxflag & WX_ATEOF)
3401         file->_flag |= _IOEOF;
3402     else if (i == -1)
3403     {
3404         file->_flag |= _IOERR;
3405         pread = 0;
3406         rcnt = 0;
3407     }
3408     if (i < 1) break;
3409   }
3410   read+=pread;
3411   _unlock_file(file);
3412   return read / size;
3413 }
3414 
3415 /*********************************************************************
3416  *		_wfreopen (MSVCRT.@)
3417  *
3418  */
3419 FILE* CDECL _wfreopen(const wchar_t *path, const wchar_t *mode, FILE* file)
3420 {
3421   int open_flags, stream_flags, fd;
3422 
3423   TRACE(":path (%p) mode (%s) file (%p) fd (%d)\n", debugstr_w(path), debugstr_w(mode), file, file ? file->_file : -1);
3424 
3425   LOCK_FILES();
3426   if (!file || ((fd = file->_file) < 0))
3427     file = NULL;
3428   else
3429   {
3430     fclose(file);
3431     /* map mode string to open() flags. "man fopen" for possibilities. */
3432     if (msvcrt_get_flags(mode, &open_flags, &stream_flags) == -1)
3433       file = NULL;
3434     else
3435     {
3436       fd = _wopen(path, open_flags, _S_IREAD | _S_IWRITE);
3437       if (fd < 0)
3438         file = NULL;
3439       else if (msvcrt_init_fp(file, fd, stream_flags) == -1)
3440       {
3441           file->_flag = 0;
3442           WARN(":failed-last error (%d)\n",GetLastError());
3443           _dosmaperr(GetLastError());
3444           file = NULL;
3445       }
3446     }
3447   }
3448   UNLOCK_FILES();
3449   return file;
3450 }
3451 
3452 /*********************************************************************
3453  *      freopen (MSVCRT.@)
3454  *
3455  */
3456 FILE* CDECL freopen(const char *path, const char *mode, FILE* file)
3457 {
3458     FILE *ret;
3459     wchar_t *pathW = NULL, *modeW = NULL;
3460 
3461     if (path && !(pathW = msvcrt_wstrdupa(path))) return NULL;
3462     if (mode && !(modeW = msvcrt_wstrdupa(mode)))
3463     {
3464         free(pathW);
3465         return NULL;
3466     }
3467 
3468     ret = _wfreopen(pathW, modeW, file);
3469 
3470     free(pathW);
3471     free(modeW);
3472     return ret;
3473 }
3474 
3475 /*********************************************************************
3476  *		fsetpos (MSVCRT.@)
3477  */
3478 int CDECL fsetpos(FILE* file, const fpos_t *pos)
3479 {
3480   int ret;
3481 
3482   _lock_file(file);
3483   /* Note that all this has been lifted 'as is' from fseek */
3484   if(file->_flag & _IOWRT)
3485 	msvcrt_flush_buffer(file);
3486 
3487   /* Discard buffered input */
3488   file->_cnt = 0;
3489   file->_ptr = file->_base;
3490 
3491   /* Reset direction of i/o */
3492   if(file->_flag & _IORW) {
3493         file->_flag &= ~(_IOREAD|_IOWRT);
3494   }
3495 
3496   ret = (_lseeki64(file->_file,*pos,SEEK_SET) == -1) ? -1 : 0;
3497   _unlock_file(file);
3498   return ret;
3499 }
3500 
3501 /*********************************************************************
3502  *		_ftelli64 (MSVCRT.@)
3503  */
3504 __int64 CDECL _ftelli64(FILE* file)
3505 {
3506     __int64 pos;
3507 
3508     _lock_file(file);
3509     pos = _telli64(file->_file);
3510     if(pos == -1) {
3511         _unlock_file(file);
3512         return -1;
3513     }
3514     if(file->_flag & (_IOMYBUF | _USERBUF))  {
3515         if(file->_flag & _IOWRT) {
3516             pos += file->_ptr - file->_base;
3517 
3518             if(get_ioinfo_nolock(file->_file)->wxflag & WX_TEXT) {
3519                 char *p;
3520 
3521                 for(p=file->_base; p<file->_ptr; p++)
3522                     if(*p == '\n')
3523                         pos++;
3524             }
3525         } else if(!file->_cnt) { /* nothing to do */
3526         } else if(_lseeki64(file->_file, 0, SEEK_END)==pos) {
3527             int i;
3528 
3529             pos -= file->_cnt;
3530             if(get_ioinfo_nolock(file->_file)->wxflag & WX_TEXT) {
3531                 for(i=0; i<file->_cnt; i++)
3532                     if(file->_ptr[i] == '\n')
3533                         pos--;
3534             }
3535         } else {
3536             char *p;
3537 
3538             if(_lseeki64(file->_file, pos, SEEK_SET) != pos) {
3539                 _unlock_file(file);
3540                 return -1;
3541             }
3542 
3543             pos -= file->_bufsiz;
3544             pos += file->_ptr - file->_base;
3545 
3546             if(get_ioinfo_nolock(file->_file)->wxflag & WX_TEXT) {
3547                 if(get_ioinfo_nolock(file->_file)->wxflag & WX_READNL)
3548                     pos--;
3549 
3550                 for(p=file->_base; p<file->_ptr; p++)
3551                     if(*p == '\n')
3552                         pos++;
3553             }
3554         }
3555     }
3556 
3557     _unlock_file(file);
3558     return pos;
3559 }
3560 
3561 /*********************************************************************
3562  *		ftell (MSVCRT.@)
3563  */
3564 LONG CDECL ftell(FILE* file)
3565 {
3566   return (LONG)_ftelli64(file);
3567 }
3568 
3569 /*********************************************************************
3570  *		fgetpos (MSVCRT.@)
3571  */
3572 int CDECL fgetpos(FILE* file, fpos_t *pos)
3573 {
3574     *pos = _ftelli64(file);
3575     if(*pos == -1)
3576         return -1;
3577     return 0;
3578 }
3579 
3580 /*********************************************************************
3581  *		fputs (MSVCRT.@)
3582  */
3583 int CDECL fputs(const char *s, FILE* file)
3584 {
3585     size_t len = strlen(s);
3586     int ret;
3587 
3588     _lock_file(file);
3589     ret = fwrite(s, sizeof(*s), len, file) == len ? 0 : EOF;
3590     _unlock_file(file);
3591     return ret;
3592 }
3593 
3594 /*********************************************************************
3595  *		fputws (MSVCRT.@)
3596  */
3597 int CDECL fputws(const wchar_t *s, FILE* file)
3598 {
3599     size_t i, len = strlenW(s);
3600     BOOL tmp_buf;
3601     int ret;
3602 
3603     _lock_file(file);
3604     if (!(get_ioinfo_nolock(file->_file)->wxflag & WX_TEXT)) {
3605         ret = fwrite(s,sizeof(*s),len,file) == len ? 0 : EOF;
3606         _unlock_file(file);
3607         return ret;
3608     }
3609 
3610     tmp_buf = add_std_buffer(file);
3611     for (i=0; i<len; i++) {
3612         if(fputwc(s[i], file) == WEOF) {
3613             if(tmp_buf) remove_std_buffer(file);
3614             _unlock_file(file);
3615             return WEOF;
3616         }
3617     }
3618 
3619     if(tmp_buf) remove_std_buffer(file);
3620     _unlock_file(file);
3621     return 0;
3622 }
3623 
3624 /*********************************************************************
3625  *		getchar (MSVCRT.@)
3626  */
3627 int CDECL getchar(void)
3628 {
3629   return fgetc(stdin);
3630 }
3631 
3632 /*********************************************************************
3633  *		getc (MSVCRT.@)
3634  */
3635 int CDECL getc(FILE* file)
3636 {
3637   return fgetc(file);
3638 }
3639 
3640 /*********************************************************************
3641  *		gets (MSVCRT.@)
3642  */
3643 char * CDECL gets(char *buf)
3644 {
3645   int    cc;
3646   char * buf_start = buf;
3647 
3648   _lock_file(stdin);
3649   for(cc = fgetc(stdin); cc != EOF && cc != '\n';
3650       cc = fgetc(stdin))
3651   if(cc != '\r') *buf++ = (char)cc;
3652 
3653   *buf = '\0';
3654 
3655   TRACE("got '%s'\n", buf_start);
3656   _unlock_file(stdin);
3657   return buf_start;
3658 }
3659 
3660 /*********************************************************************
3661  *		_getws (MSVCRT.@)
3662  */
3663 wchar_t* CDECL _getws(wchar_t* buf)
3664 {
3665     wint_t cc;
3666     wchar_t* ws = buf;
3667 
3668     _lock_file(stdin);
3669     for (cc = fgetwc(stdin); cc != WEOF && cc != '\n';
3670          cc = fgetwc(stdin))
3671     {
3672         if (cc != '\r')
3673             *buf++ = (wchar_t)cc;
3674     }
3675     *buf = '\0';
3676 
3677     TRACE("got %s\n", debugstr_w(ws));
3678     _unlock_file(stdin);
3679     return ws;
3680 }
3681 
3682 /*********************************************************************
3683  *		putc (MSVCRT.@)
3684  */
3685 int CDECL putc(int c, FILE* file)
3686 {
3687   return fputc(c, file);
3688 }
3689 
3690 /*********************************************************************
3691  *		putchar (MSVCRT.@)
3692  */
3693 int CDECL putchar(int c)
3694 {
3695   return fputc(c, stdout);
3696 }
3697 
3698 /*********************************************************************
3699  *		_putwch (MSVCRT.@)
3700  */
3701 wint_t CDECL _putwch(wchar_t c)
3702 {
3703   return fputwc(c, stdout);
3704 }
3705 
3706 /*********************************************************************
3707  *		puts (MSVCRT.@)
3708  */
3709 int CDECL puts(const char *s)
3710 {
3711     size_t len = strlen(s);
3712     int ret;
3713 
3714     _lock_file(stdout);
3715     if(fwrite(s, sizeof(*s), len, stdout) != len) {
3716         _unlock_file(stdout);
3717         return EOF;
3718     }
3719 
3720     ret = fwrite("\n",1,1,stdout) == 1 ? 0 : EOF;
3721     _unlock_file(stdout);
3722     return ret;
3723 }
3724 
3725 /*********************************************************************
3726  *		_putws (MSVCRT.@)
3727  */
3728 int CDECL _putws(const wchar_t *s)
3729 {
3730     static const wchar_t nl = '\n';
3731     size_t len = strlenW(s);
3732     int ret;
3733 
3734     _lock_file(stdout);
3735     if(fwrite(s, sizeof(*s), len, stdout) != len) {
3736         _unlock_file(stdout);
3737         return EOF;
3738     }
3739 
3740     ret = fwrite(&nl,sizeof(nl),1,stdout) == 1 ? 0 : EOF;
3741     _unlock_file(stdout);
3742     return ret;
3743 }
3744 
3745 /*********************************************************************
3746  *		remove (MSVCRT.@)
3747  */
3748 int CDECL remove(const char *path)
3749 {
3750   TRACE("(%s)\n",path);
3751   if (DeleteFileA(path))
3752     return 0;
3753   TRACE(":failed (%d)\n",GetLastError());
3754   _dosmaperr(GetLastError());
3755   return -1;
3756 }
3757 
3758 /*********************************************************************
3759  *		_wremove (MSVCRT.@)
3760  */
3761 int CDECL _wremove(const wchar_t *path)
3762 {
3763   TRACE("(%s)\n",debugstr_w(path));
3764   if (DeleteFileW(path))
3765     return 0;
3766   TRACE(":failed (%d)\n",GetLastError());
3767   _dosmaperr(GetLastError());
3768   return -1;
3769 }
3770 
3771 /*********************************************************************
3772  *		rename (MSVCRT.@)
3773  */
3774 int CDECL rename(const char *oldpath,const char *newpath)
3775 {
3776   TRACE(":from %s to %s\n",oldpath,newpath);
3777   if (MoveFileExA(oldpath, newpath, MOVEFILE_COPY_ALLOWED))
3778     return 0;
3779   TRACE(":failed (%d)\n",GetLastError());
3780   _dosmaperr(GetLastError());
3781   return -1;
3782 }
3783 
3784 /*********************************************************************
3785  *		_wrename (MSVCRT.@)
3786  */
3787 int CDECL _wrename(const wchar_t *oldpath,const wchar_t *newpath)
3788 {
3789   TRACE(":from %s to %s\n",debugstr_w(oldpath),debugstr_w(newpath));
3790   if (MoveFileExW(oldpath, newpath, MOVEFILE_COPY_ALLOWED))
3791     return 0;
3792   TRACE(":failed (%d)\n",GetLastError());
3793   _dosmaperr(GetLastError());
3794   return -1;
3795 }
3796 
3797 /*********************************************************************
3798  *		setvbuf (MSVCRT.@)
3799  */
3800 int CDECL setvbuf(FILE* file, char *buf, int mode, size_t size)
3801 {
3802     if(!MSVCRT_CHECK_PMT(file != NULL)) return -1;
3803     if(!MSVCRT_CHECK_PMT(mode==_IONBF || mode==_IOFBF || mode==_IOLBF)) return -1;
3804     if(!MSVCRT_CHECK_PMT(mode==_IONBF || (size>=2 && size<=INT_MAX))) return -1;
3805 
3806     _lock_file(file);
3807 
3808     fflush(file);
3809     if(file->_flag & _IOMYBUF)
3810         free(file->_base);
3811     file->_flag &= ~(_IONBF | _IOMYBUF | _USERBUF);
3812     file->_cnt = 0;
3813 
3814     if(mode == _IONBF) {
3815         file->_flag |= _IONBF;
3816         file->_base = file->_ptr = (char*)&file->_charbuf;
3817         file->_bufsiz = 2;
3818     }else if(buf) {
3819         file->_base = file->_ptr = buf;
3820         file->_flag |= _USERBUF;
3821         file->_bufsiz = size;
3822     }else {
3823         file->_base = file->_ptr = malloc(size);
3824         if(!file->_base) {
3825             file->_bufsiz = 0;
3826             _unlock_file(file);
3827             return -1;
3828         }
3829 
3830         file->_flag |= _IOMYBUF;
3831         file->_bufsiz = size;
3832     }
3833     _unlock_file(file);
3834     return 0;
3835 }
3836 
3837 /*********************************************************************
3838  *		setbuf (MSVCRT.@)
3839  */
3840 void CDECL setbuf(FILE* file, char *buf)
3841 {
3842   setvbuf(file, buf, buf ? _IOFBF : _IONBF, BUFSIZ);
3843 }
3844 
3845 /*********************************************************************
3846  *		tmpnam (MSVCRT.@)
3847  */
3848 char * CDECL tmpnam(char *s)
3849 {
3850   char tmpstr[16];
3851   char *p;
3852   int count, size;
3853 
3854   if (!s) {
3855     thread_data_t *data = msvcrt_get_thread_data();
3856 
3857     if(!data->tmpnam_buffer)
3858       data->tmpnam_buffer = malloc(MAX_PATH);
3859 
3860     s = data->tmpnam_buffer;
3861   }
3862 
3863   msvcrt_int_to_base32(GetCurrentProcessId(), tmpstr);
3864   p = s + sprintf(s, "\\s%s.", tmpstr);
3865   for (count = 0; count < TMP_MAX; count++)
3866   {
3867     size = msvcrt_int_to_base32(tmpnam_unique++, tmpstr);
3868     memcpy(p, tmpstr, size);
3869     p[size] = '\0';
3870     if (GetFileAttributesA(s) == INVALID_FILE_ATTRIBUTES &&
3871         GetLastError() == ERROR_FILE_NOT_FOUND)
3872       break;
3873   }
3874   return s;
3875 }
3876 
3877 /*********************************************************************
3878  *              _wtmpnam (MSVCRT.@)
3879  */
3880 wchar_t * CDECL _wtmpnam(wchar_t *s)
3881 {
3882     static const wchar_t format[] = {'\\','s','%','s','.',0};
3883     wchar_t tmpstr[16];
3884     wchar_t *p;
3885     int count, size;
3886     if (!s) {
3887         thread_data_t *data = msvcrt_get_thread_data();
3888 
3889         if(!data->wtmpnam_buffer)
3890             data->wtmpnam_buffer = malloc(sizeof(wchar_t[MAX_PATH]));
3891 
3892         s = data->wtmpnam_buffer;
3893     }
3894 
3895     msvcrt_int_to_base32_w(GetCurrentProcessId(), tmpstr);
3896     p = s + _snwprintf(s, MAX_PATH, format, tmpstr);
3897     for (count = 0; count < TMP_MAX; count++)
3898     {
3899         size = msvcrt_int_to_base32_w(tmpnam_unique++, tmpstr);
3900         memcpy(p, tmpstr, size*sizeof(wchar_t));
3901         p[size] = '\0';
3902         if (GetFileAttributesW(s) == INVALID_FILE_ATTRIBUTES &&
3903                 GetLastError() == ERROR_FILE_NOT_FOUND)
3904             break;
3905     }
3906     return s;
3907 }
3908 
3909 /*********************************************************************
3910  *		tmpfile (MSVCRT.@)
3911  */
3912 FILE* CDECL tmpfile(void)
3913 {
3914   char *filename = tmpnam(NULL);
3915   int fd;
3916   FILE* file = NULL;
3917 
3918   LOCK_FILES();
3919   fd = _open(filename, _O_CREAT | _O_BINARY | _O_RDWR | _O_TEMPORARY,
3920           _S_IREAD | _S_IWRITE);
3921   if (fd != -1 && (file = msvcrt_alloc_fp()))
3922   {
3923     if (msvcrt_init_fp(file, fd, _IORW) == -1)
3924     {
3925         file->_flag = 0;
3926         file = NULL;
3927     }
3928     else file->_tmpfname = _strdup(filename);
3929   }
3930 
3931   if(fd != -1 && !file)
3932       _close(fd);
3933   UNLOCK_FILES();
3934   return file;
3935 }
3936 
3937 /*********************************************************************
3938  *		ungetc (MSVCRT.@)
3939  */
3940 int CDECL ungetc(int c, FILE * file)
3941 {
3942     if(!MSVCRT_CHECK_PMT(file != NULL)) return EOF;
3943 
3944     if (c == EOF || !(file->_flag&_IOREAD ||
3945                 (file->_flag&_IORW && !(file->_flag&_IOWRT))))
3946         return EOF;
3947 
3948     _lock_file(file);
3949     if((!(file->_flag & (_IONBF | _IOMYBUF | _USERBUF))
3950                 && msvcrt_alloc_buffer(file))
3951             || (!file->_cnt && file->_ptr==file->_base))
3952         file->_ptr++;
3953 
3954     if(file->_ptr>file->_base) {
3955         file->_ptr--;
3956         if(file->_flag & _IOSTRG) {
3957             if(*file->_ptr != c) {
3958                 file->_ptr++;
3959                 _unlock_file(file);
3960                 return EOF;
3961             }
3962         }else {
3963             *file->_ptr = c;
3964         }
3965         file->_cnt++;
3966         clearerr(file);
3967         file->_flag |= _IOREAD;
3968         _unlock_file(file);
3969         return c;
3970     }
3971 
3972     _unlock_file(file);
3973     return EOF;
3974 }
3975 
3976 /*********************************************************************
3977  *              ungetwc (MSVCRT.@)
3978  */
3979 wint_t CDECL ungetwc(wint_t wc, FILE * file)
3980 {
3981     wchar_t mwc = wc;
3982 
3983     if (wc == WEOF)
3984         return WEOF;
3985 
3986     _lock_file(file);
3987 
3988     if((get_ioinfo_nolock(file->_file)->exflag & (EF_UTF8 | EF_UTF16))
3989             || !(get_ioinfo_nolock(file->_file)->wxflag & WX_TEXT)) {
3990         unsigned char * pp = (unsigned char *)&mwc;
3991         int i;
3992 
3993         for(i=sizeof(wchar_t)-1;i>=0;i--) {
3994             if(pp[i] != ungetc(pp[i],file)) {
3995                 _unlock_file(file);
3996                 return WEOF;
3997             }
3998         }
3999     }else {
4000         char mbs[MB_LEN_MAX];
4001         int len;
4002 
4003         len = wctomb(mbs, mwc);
4004         if(len == -1) {
4005             _unlock_file(file);
4006             return WEOF;
4007         }
4008 
4009         for(len--; len>=0; len--) {
4010             if(mbs[len] != ungetc(mbs[len], file)) {
4011                 _unlock_file(file);
4012                 return WEOF;
4013             }
4014         }
4015     }
4016 
4017     _unlock_file(file);
4018     return mwc;
4019 }
4020 
4021 
4022 
4023 /*********************************************************************
4024  *		_getmaxstdio (MSVCRT.@)
4025  */
4026 int CDECL _getmaxstdio(void)
4027 {
4028     return MSVCRT_max_streams;
4029 }
4030 
4031 /*********************************************************************
4032  *		_setmaxstdio (MSVCRT.@)
4033  */
4034 int CDECL _setmaxstdio(int newmax)
4035 {
4036     TRACE("%d\n", newmax);
4037 
4038     if(newmax<_IOB_ENTRIES || newmax>MSVCRT_MAX_FILES || newmax<MSVCRT_stream_idx)
4039         return -1;
4040 
4041     MSVCRT_max_streams = newmax;
4042     return MSVCRT_max_streams;
4043 }
4044