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_nolock(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 wxflag_ptr++; handle_ptr++; 449 } 450 return TRUE; 451 } 452 453 /* INTERNAL: Set up all file descriptors, 454 * as well as default streams (stdin, stderr and stdout) 455 */ 456 void msvcrt_init_io(void) 457 { 458 STARTUPINFOA si; 459 unsigned int i; 460 ioinfo *fdinfo; 461 462 GetStartupInfoA(&si); 463 if (si.cbReserved2 >= sizeof(unsigned int) && si.lpReserved2 != NULL) 464 { 465 BYTE* wxflag_ptr; 466 HANDLE* handle_ptr; 467 unsigned int count; 468 469 count = *(unsigned*)si.lpReserved2; 470 wxflag_ptr = si.lpReserved2 + sizeof(unsigned); 471 handle_ptr = (HANDLE*)(wxflag_ptr + count); 472 473 count = min(count, (si.cbReserved2 - sizeof(unsigned)) / (sizeof(HANDLE) + 1)); 474 count = min(count, MSVCRT_MAX_FILES); 475 for (i = 0; i < count; i++) 476 { 477 if ((*wxflag_ptr & WX_OPEN) && GetFileType(*handle_ptr) != FILE_TYPE_UNKNOWN) 478 { 479 fdinfo = get_ioinfo_alloc_fd(i); 480 if(fdinfo != &__badioinfo) 481 msvcrt_set_fd(fdinfo, *handle_ptr, *wxflag_ptr); 482 release_ioinfo(fdinfo); 483 } 484 485 wxflag_ptr++; handle_ptr++; 486 } 487 } 488 489 fdinfo = get_ioinfo_alloc_fd(STDIN_FILENO); 490 if (!(fdinfo->wxflag & WX_OPEN) || fdinfo->handle == INVALID_HANDLE_VALUE) { 491 HANDLE h = GetStdHandle(STD_INPUT_HANDLE); 492 DWORD type = GetFileType(h); 493 494 msvcrt_set_fd(fdinfo, h, WX_OPEN|WX_TEXT|((type&0xf)==FILE_TYPE_CHAR ? WX_TTY : 0) 495 |((type&0xf)==FILE_TYPE_PIPE ? WX_PIPE : 0)); 496 } 497 release_ioinfo(fdinfo); 498 499 fdinfo = get_ioinfo_alloc_fd(STDOUT_FILENO); 500 if (!(fdinfo->wxflag & WX_OPEN) || fdinfo->handle == INVALID_HANDLE_VALUE) { 501 HANDLE h = GetStdHandle(STD_OUTPUT_HANDLE); 502 DWORD type = GetFileType(h); 503 504 msvcrt_set_fd(fdinfo, h, WX_OPEN|WX_TEXT|((type&0xf)==FILE_TYPE_CHAR ? WX_TTY : 0) 505 |((type&0xf)==FILE_TYPE_PIPE ? WX_PIPE : 0)); 506 } 507 release_ioinfo(fdinfo); 508 509 fdinfo = get_ioinfo_alloc_fd(STDERR_FILENO); 510 if (!(fdinfo->wxflag & WX_OPEN) || fdinfo->handle == INVALID_HANDLE_VALUE) { 511 HANDLE h = GetStdHandle(STD_ERROR_HANDLE); 512 DWORD type = GetFileType(h); 513 514 msvcrt_set_fd(fdinfo, h, WX_OPEN|WX_TEXT|((type&0xf)==FILE_TYPE_CHAR ? WX_TTY : 0) 515 |((type&0xf)==FILE_TYPE_PIPE ? WX_PIPE : 0)); 516 } 517 release_ioinfo(fdinfo); 518 519 TRACE(":handles (%p)(%p)(%p)\n", get_ioinfo_nolock(STDIN_FILENO)->handle, 520 get_ioinfo_nolock(STDOUT_FILENO)->handle, 521 get_ioinfo_nolock(STDERR_FILENO)->handle); 522 523 memset(_iob,0,3*sizeof(FILE)); 524 for (i = 0; i < 3; i++) 525 { 526 /* FILE structs for stdin/out/err are static and never deleted */ 527 _iob[i]._file = i; 528 _iob[i]._tmpfname = NULL; 529 _iob[i]._flag = (i == 0) ? _IOREAD : _IOWRT; 530 } 531 MSVCRT_stream_idx = 3; 532 } 533 534 /* INTERNAL: Flush stdio file buffer */ 535 static int msvcrt_flush_buffer(FILE* file) 536 { 537 if((file->_flag & (_IOREAD|_IOWRT)) == _IOWRT && 538 file->_flag & (_IOMYBUF|_USERBUF)) { 539 int cnt=file->_ptr-file->_base; 540 if(cnt>0 && _write(file->_file, file->_base, cnt) != cnt) { 541 file->_flag |= _IOERR; 542 return EOF; 543 } 544 545 if(file->_flag & _IORW) 546 file->_flag &= ~_IOWRT; 547 548 #ifdef __REACTOS__ /* CORE-11949 */ 549 file->_ptr=file->_base; 550 file->_cnt=0; 551 #endif 552 } 553 554 #ifndef __REACTOS__ /* CORE-11949 */ 555 file->_ptr=file->_base; 556 file->_cnt=0; 557 #endif 558 return 0; 559 } 560 561 /********************************************************************* 562 * _isatty (MSVCRT.@) 563 */ 564 int CDECL _isatty(int fd) 565 { 566 TRACE(":fd (%d)\n",fd); 567 568 return get_ioinfo_nolock(fd)->wxflag & WX_TTY; 569 } 570 571 /* INTERNAL: Allocate stdio file buffer */ 572 /*static*/ BOOL msvcrt_alloc_buffer(FILE* file) 573 { 574 if((file->_file==STDOUT_FILENO || file->_file==STDERR_FILENO) 575 && _isatty(file->_file)) 576 return FALSE; 577 578 file->_base = calloc(1, MSVCRT_INTERNAL_BUFSIZ); 579 if(file->_base) { 580 file->_bufsiz = MSVCRT_INTERNAL_BUFSIZ; 581 file->_flag |= _IOMYBUF; 582 } else { 583 file->_base = (char*)(&file->_charbuf); 584 file->_bufsiz = 2; 585 file->_flag |= _IONBF; 586 } 587 file->_ptr = file->_base; 588 file->_cnt = 0; 589 return TRUE; 590 } 591 592 /* INTERNAL: Allocate temporary buffer for stdout and stderr */ 593 static BOOL add_std_buffer(FILE *file) 594 { 595 static char buffers[2][BUFSIZ]; 596 597 if((file->_file!=STDOUT_FILENO && file->_file!=STDERR_FILENO) 598 || (file->_flag & (_IONBF | _IOMYBUF | _USERBUF)) 599 || !_isatty(file->_file)) 600 return FALSE; 601 602 file->_ptr = file->_base = buffers[file->_file == STDOUT_FILENO ? 0 : 1]; 603 file->_bufsiz = file->_cnt = BUFSIZ; 604 file->_flag |= _USERBUF; 605 return TRUE; 606 } 607 608 /* INTERNAL: Removes temporary buffer from stdout or stderr */ 609 /* Only call this function when add_std_buffer returned TRUE */ 610 static void remove_std_buffer(FILE *file) 611 { 612 msvcrt_flush_buffer(file); 613 file->_ptr = file->_base = NULL; 614 file->_bufsiz = file->_cnt = 0; 615 file->_flag &= ~_USERBUF; 616 } 617 618 /* INTERNAL: Convert integer to base32 string (0-9a-v), 0 becomes "" */ 619 static int msvcrt_int_to_base32(int num, char *str) 620 { 621 char *p; 622 int n = num; 623 int digits = 0; 624 625 while (n != 0) 626 { 627 n >>= 5; 628 digits++; 629 } 630 p = str + digits; 631 *p = 0; 632 while (--p >= str) 633 { 634 *p = (num & 31) + '0'; 635 if (*p > '9') 636 *p += ('a' - '0' - 10); 637 num >>= 5; 638 } 639 640 return digits; 641 } 642 643 /* INTERNAL: wide character version of msvcrt_int_to_base32 */ 644 static int msvcrt_int_to_base32_w(int num, wchar_t *str) 645 { 646 wchar_t *p; 647 int n = num; 648 int digits = 0; 649 650 while (n != 0) 651 { 652 n >>= 5; 653 digits++; 654 } 655 p = str + digits; 656 *p = 0; 657 while (--p >= str) 658 { 659 *p = (num & 31) + '0'; 660 if (*p > '9') 661 *p += ('a' - '0' - 10); 662 num >>= 5; 663 } 664 665 return digits; 666 } 667 668 /* INTERNAL: Create a wide string from an ascii string */ 669 wchar_t *msvcrt_wstrdupa(const char *str) 670 { 671 const unsigned int len = strlen(str) + 1 ; 672 wchar_t *wstr = malloc(len* sizeof (wchar_t)); 673 if (!wstr) 674 return NULL; 675 MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED,str,len,wstr,len); 676 return wstr; 677 } 678 679 /********************************************************************* 680 * __iob_func(MSVCRT.@) 681 */ 682 FILE * CDECL __iob_func(void) 683 { 684 return &_iob[0]; 685 } 686 687 /********************************************************************* 688 * _access (MSVCRT.@) 689 */ 690 int CDECL _access(const char *filename, int mode) 691 { 692 DWORD attr = GetFileAttributesA(filename); 693 694 TRACE("(%s,%d) %d\n",filename,mode,attr); 695 696 if (!filename || attr == INVALID_FILE_ATTRIBUTES) 697 { 698 _dosmaperr(GetLastError()); 699 return -1; 700 } 701 if ((attr & FILE_ATTRIBUTE_READONLY) && (mode & W_OK)) 702 { 703 _set_errno(ERROR_ACCESS_DENIED); 704 return -1; 705 } 706 return 0; 707 } 708 709 /********************************************************************* 710 * _access_s (MSVCRT.@) 711 */ 712 int CDECL _access_s(const char *filename, int mode) 713 { 714 if (!MSVCRT_CHECK_PMT(filename != NULL) || 715 !MSVCRT_CHECK_PMT((mode & ~(R_OK | W_OK)) == 0)) 716 { 717 _set_errno(EINVAL); 718 return -1; 719 } 720 721 return _access(filename, mode); 722 } 723 724 /********************************************************************* 725 * _waccess (MSVCRT.@) 726 */ 727 int CDECL _waccess(const wchar_t *filename, int mode) 728 { 729 DWORD attr = GetFileAttributesW(filename); 730 731 TRACE("(%s,%d) %d\n",debugstr_w(filename),mode,attr); 732 733 if (!filename || attr == INVALID_FILE_ATTRIBUTES) 734 { 735 _dosmaperr(GetLastError()); 736 return -1; 737 } 738 if ((attr & FILE_ATTRIBUTE_READONLY) && (mode & W_OK)) 739 { 740 _set_errno(ERROR_ACCESS_DENIED); 741 return -1; 742 } 743 return 0; 744 } 745 746 /********************************************************************* 747 * _waccess_s (MSVCRT.@) 748 */ 749 int CDECL _waccess_s(const wchar_t *filename, int mode) 750 { 751 if (!MSVCRT_CHECK_PMT(filename != NULL) || 752 !MSVCRT_CHECK_PMT((mode & ~(R_OK | W_OK)) == 0)) 753 { 754 *_errno() = EINVAL; 755 return -1; 756 } 757 758 return _waccess(filename, mode); 759 } 760 761 /********************************************************************* 762 * _chmod (MSVCRT.@) 763 */ 764 int CDECL _chmod(const char *path, int flags) 765 { 766 DWORD oldFlags = GetFileAttributesA(path); 767 768 if (oldFlags != INVALID_FILE_ATTRIBUTES) 769 { 770 DWORD newFlags = (flags & _S_IWRITE)? oldFlags & ~FILE_ATTRIBUTE_READONLY: 771 oldFlags | FILE_ATTRIBUTE_READONLY; 772 773 if (newFlags == oldFlags || SetFileAttributesA(path, newFlags)) 774 return 0; 775 } 776 _dosmaperr(GetLastError()); 777 return -1; 778 } 779 780 /********************************************************************* 781 * _wchmod (MSVCRT.@) 782 */ 783 int CDECL _wchmod(const wchar_t *path, int flags) 784 { 785 DWORD oldFlags = GetFileAttributesW(path); 786 787 if (oldFlags != INVALID_FILE_ATTRIBUTES) 788 { 789 DWORD newFlags = (flags & _S_IWRITE)? oldFlags & ~FILE_ATTRIBUTE_READONLY: 790 oldFlags | FILE_ATTRIBUTE_READONLY; 791 792 if (newFlags == oldFlags || SetFileAttributesW(path, newFlags)) 793 return 0; 794 } 795 _dosmaperr(GetLastError()); 796 return -1; 797 } 798 799 /********************************************************************* 800 * _unlink (MSVCRT.@) 801 */ 802 int CDECL _unlink(const char *path) 803 { 804 TRACE("%s\n",debugstr_a(path)); 805 if(DeleteFileA(path)) 806 return 0; 807 TRACE("failed (%d)\n",GetLastError()); 808 _dosmaperr(GetLastError()); 809 return -1; 810 } 811 812 /********************************************************************* 813 * _wunlink (MSVCRT.@) 814 */ 815 int CDECL _wunlink(const wchar_t *path) 816 { 817 TRACE("(%s)\n",debugstr_w(path)); 818 if(DeleteFileW(path)) 819 return 0; 820 TRACE("failed (%d)\n",GetLastError()); 821 _dosmaperr(GetLastError()); 822 return -1; 823 } 824 825 /********************************************************************* 826 * _commit (MSVCRT.@) 827 */ 828 int CDECL _commit(int fd) 829 { 830 ioinfo *info = get_ioinfo(fd); 831 int ret; 832 833 TRACE(":fd (%d) handle (%p)\n", fd, info->handle); 834 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 & ~WX_DONTINHERIT; 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 if (!MSVCRT_CHECK_PMT(file != NULL)) return EOF; 2788 2789 _lock_file(file); 2790 flag = file->_flag; 2791 free(file->_tmpfname); 2792 file->_tmpfname = NULL; 2793 /* flush stdio buffers */ 2794 if(file->_flag & _IOWRT) 2795 fflush(file); 2796 if(file->_flag & _IOMYBUF) 2797 free(file->_base); 2798 2799 r=_close(file->_file); 2800 2801 file->_flag = 0; 2802 _unlock_file(file); 2803 if(file<_iob || file>=_iob+_IOB_ENTRIES) 2804 DeleteCriticalSection(&((file_crit*)file)->crit); 2805 2806 if(file == msvcrt_get_file(MSVCRT_stream_idx-1)) { 2807 while(MSVCRT_stream_idx>3 && !file->_flag) { 2808 MSVCRT_stream_idx--; 2809 file = msvcrt_get_file(MSVCRT_stream_idx-1); 2810 } 2811 } 2812 2813 return ((r == -1) || (flag & _IOERR) ? EOF : 0); 2814 } 2815 2816 /********************************************************************* 2817 * feof (MSVCRT.@) 2818 */ 2819 int CDECL feof(FILE* file) 2820 { 2821 return file->_flag & _IOEOF; 2822 } 2823 2824 /********************************************************************* 2825 * ferror (MSVCRT.@) 2826 */ 2827 int CDECL ferror(FILE* file) 2828 { 2829 return file->_flag & _IOERR; 2830 } 2831 2832 /********************************************************************* 2833 * _filbuf (MSVCRT.@) 2834 */ 2835 int CDECL _filbuf(FILE* file) 2836 { 2837 unsigned char c; 2838 _lock_file(file); 2839 2840 if(file->_flag & _IOSTRG) { 2841 _unlock_file(file); 2842 return EOF; 2843 } 2844 2845 /* Allocate buffer if needed */ 2846 if(!(file->_flag & (_IONBF | _IOMYBUF | _USERBUF))) 2847 msvcrt_alloc_buffer(file); 2848 2849 if(!(file->_flag & _IOREAD)) { 2850 if(file->_flag & _IORW) 2851 file->_flag |= _IOREAD; 2852 else { 2853 _unlock_file(file); 2854 return EOF; 2855 } 2856 } 2857 2858 if(!(file->_flag & (_IOMYBUF | _USERBUF))) { 2859 int r; 2860 if ((r = _read(file->_file,&c,1)) != 1) { 2861 file->_flag |= (r == 0) ? _IOEOF : _IOERR; 2862 _unlock_file(file); 2863 return EOF; 2864 } 2865 2866 _unlock_file(file); 2867 return c; 2868 } else { 2869 file->_cnt = _read(file->_file, file->_base, file->_bufsiz); 2870 if(file->_cnt<=0) { 2871 file->_flag |= (file->_cnt == 0) ? _IOEOF : _IOERR; 2872 file->_cnt = 0; 2873 _unlock_file(file); 2874 return EOF; 2875 } 2876 2877 file->_cnt--; 2878 file->_ptr = file->_base+1; 2879 c = *(unsigned char *)file->_base; 2880 _unlock_file(file); 2881 return c; 2882 } 2883 } 2884 2885 /********************************************************************* 2886 * fgetc (MSVCRT.@) 2887 */ 2888 int CDECL fgetc(FILE* file) 2889 { 2890 unsigned char *i; 2891 unsigned int j; 2892 2893 _lock_file(file); 2894 if (file->_cnt>0) { 2895 file->_cnt--; 2896 i = (unsigned char *)file->_ptr++; 2897 j = *i; 2898 } else 2899 j = _filbuf(file); 2900 2901 _unlock_file(file); 2902 return j; 2903 } 2904 2905 /********************************************************************* 2906 * _fgetchar (MSVCRT.@) 2907 */ 2908 int CDECL _fgetchar(void) 2909 { 2910 return fgetc(stdin); 2911 } 2912 2913 /********************************************************************* 2914 * fgets (MSVCRT.@) 2915 */ 2916 char * CDECL fgets(char *s, int size, FILE* file) 2917 { 2918 int cc = EOF; 2919 char * buf_start = s; 2920 2921 TRACE(":file(%p) fd (%d) str (%p) len (%d)\n", 2922 file,file->_file,s,size); 2923 2924 _lock_file(file); 2925 2926 while ((size >1) && (cc = fgetc(file)) != EOF && cc != '\n') 2927 { 2928 *s++ = (char)cc; 2929 size --; 2930 } 2931 if ((cc == EOF) && (s == buf_start)) /* If nothing read, return 0*/ 2932 { 2933 TRACE(":nothing read\n"); 2934 _unlock_file(file); 2935 return NULL; 2936 } 2937 if ((cc != EOF) && (size > 1)) 2938 *s++ = cc; 2939 *s = '\0'; 2940 TRACE(":got %s\n", debugstr_a(buf_start)); 2941 _unlock_file(file); 2942 return buf_start; 2943 } 2944 2945 /********************************************************************* 2946 * fgetwc (MSVCRT.@) 2947 */ 2948 wint_t CDECL fgetwc(FILE* file) 2949 { 2950 wint_t ret; 2951 int ch; 2952 2953 _lock_file(file); 2954 2955 if((get_ioinfo_nolock(file->_file)->exflag & (EF_UTF8 | EF_UTF16)) 2956 || !(get_ioinfo_nolock(file->_file)->wxflag & WX_TEXT)) { 2957 char *p; 2958 2959 for(p=(char*)&ret; (wint_t*)p<&ret+1; p++) { 2960 ch = fgetc(file); 2961 if(ch == EOF) { 2962 ret = WEOF; 2963 break; 2964 } 2965 *p = (char)ch; 2966 } 2967 }else { 2968 char mbs[MB_LEN_MAX]; 2969 int len = 0; 2970 2971 ch = fgetc(file); 2972 if(ch != EOF) { 2973 mbs[0] = (char)ch; 2974 if(isleadbyte((unsigned char)mbs[0])) { 2975 ch = fgetc(file); 2976 if(ch != EOF) { 2977 mbs[1] = (char)ch; 2978 len = 2; 2979 } 2980 }else { 2981 len = 1; 2982 } 2983 } 2984 2985 if(!len || mbtowc(&ret, mbs, len)==-1) 2986 ret = WEOF; 2987 } 2988 2989 _unlock_file(file); 2990 return ret; 2991 } 2992 2993 /********************************************************************* 2994 * _getw (MSVCRT.@) 2995 */ 2996 int CDECL _getw(FILE* file) 2997 { 2998 char *ch; 2999 int i, k; 3000 unsigned int j; 3001 ch = (char *)&i; 3002 3003 _lock_file(file); 3004 for (j=0; j<sizeof(int); j++) { 3005 k = fgetc(file); 3006 if (k == EOF) { 3007 file->_flag |= _IOEOF; 3008 _unlock_file(file); 3009 return EOF; 3010 } 3011 ch[j] = k; 3012 } 3013 3014 _unlock_file(file); 3015 return i; 3016 } 3017 3018 /********************************************************************* 3019 * getwc (MSVCRT.@) 3020 */ 3021 wint_t CDECL getwc(FILE* file) 3022 { 3023 return fgetwc(file); 3024 } 3025 3026 /********************************************************************* 3027 * _fgetwchar (MSVCRT.@) 3028 */ 3029 wint_t CDECL _fgetwchar(void) 3030 { 3031 return fgetwc(stdin); 3032 } 3033 3034 /********************************************************************* 3035 * getwchar (MSVCRT.@) 3036 */ 3037 wint_t CDECL getwchar(void) 3038 { 3039 return _fgetwchar(); 3040 } 3041 3042 /********************************************************************* 3043 * fgetws (MSVCRT.@) 3044 */ 3045 wchar_t * CDECL fgetws(wchar_t *s, int size, FILE* file) 3046 { 3047 int cc = WEOF; 3048 wchar_t * buf_start = s; 3049 3050 TRACE(":file(%p) fd (%d) str (%p) len (%d)\n", 3051 file,file->_file,s,size); 3052 3053 _lock_file(file); 3054 3055 while ((size >1) && (cc = fgetwc(file)) != WEOF && cc != '\n') 3056 { 3057 *s++ = (char)cc; 3058 size --; 3059 } 3060 if ((cc == WEOF) && (s == buf_start)) /* If nothing read, return 0*/ 3061 { 3062 TRACE(":nothing read\n"); 3063 _unlock_file(file); 3064 return NULL; 3065 } 3066 if ((cc != WEOF) && (size > 1)) 3067 *s++ = cc; 3068 *s = 0; 3069 TRACE(":got %s\n", debugstr_w(buf_start)); 3070 _unlock_file(file); 3071 return buf_start; 3072 } 3073 3074 /********************************************************************* 3075 * fwrite (MSVCRT.@) 3076 */ 3077 size_t CDECL fwrite(const void *ptr, size_t size, size_t nmemb, FILE* file) 3078 { 3079 size_t wrcnt=size * nmemb; 3080 int written = 0; 3081 if (size == 0) 3082 return 0; 3083 3084 _lock_file(file); 3085 3086 while(wrcnt) { 3087 #ifndef __REACTOS__ 3088 if(file->_cnt < 0) { 3089 WARN("negative file->_cnt value in %p\n", file); 3090 file->_flag |= MSVCRT__IOERR; 3091 break; 3092 } else 3093 #endif 3094 if(file->_cnt) { 3095 int pcnt=(file->_cnt>wrcnt)? wrcnt: file->_cnt; 3096 memcpy(file->_ptr, ptr, pcnt); 3097 file->_cnt -= pcnt; 3098 file->_ptr += pcnt; 3099 written += pcnt; 3100 wrcnt -= pcnt; 3101 ptr = (const char*)ptr + pcnt; 3102 } else if((file->_flag & _IONBF) 3103 || ((file->_flag & (_IOMYBUF | _USERBUF)) && wrcnt >= file->_bufsiz) 3104 || (!(file->_flag & (_IOMYBUF | _USERBUF)) && wrcnt >= MSVCRT_INTERNAL_BUFSIZ)) { 3105 size_t pcnt; 3106 int bufsiz; 3107 3108 if(file->_flag & _IONBF) 3109 bufsiz = 1; 3110 else if(!(file->_flag & (_IOMYBUF | _USERBUF))) 3111 bufsiz = MSVCRT_INTERNAL_BUFSIZ; 3112 else 3113 bufsiz = file->_bufsiz; 3114 3115 pcnt = (wrcnt / bufsiz) * bufsiz; 3116 3117 if(msvcrt_flush_buffer(file) == EOF) 3118 break; 3119 3120 if(_write(file->_file, ptr, pcnt) <= 0) { 3121 file->_flag |= _IOERR; 3122 break; 3123 } 3124 written += pcnt; 3125 wrcnt -= pcnt; 3126 ptr = (const char*)ptr + pcnt; 3127 } else { 3128 if(_flsbuf(*(const char*)ptr, file) == EOF) 3129 break; 3130 written++; 3131 wrcnt--; 3132 ptr = (const char*)ptr + 1; 3133 } 3134 } 3135 3136 _unlock_file(file); 3137 return written / size; 3138 } 3139 3140 /********************************************************************* 3141 * fputwc (MSVCRT.@) 3142 * FORKED for ReactOS, don't sync with Wine! 3143 * References: 3144 * - http://jira.reactos.org/browse/CORE-6495 3145 * - http://bugs.winehq.org/show_bug.cgi?id=8598 3146 */ 3147 wint_t CDECL fputwc(wchar_t c, FILE* stream) 3148 { 3149 /* If this is a real file stream (and not some temporary one for 3150 sprintf-like functions), check whether it is opened in text mode. 3151 In this case, we have to perform an implicit conversion to ANSI. */ 3152 if (!(stream->_flag & _IOSTRG) && get_ioinfo_nolock(stream->_file)->wxflag & WX_TEXT) 3153 { 3154 /* Convert to multibyte in text mode */ 3155 char mbc[MB_LEN_MAX]; 3156 int mb_return; 3157 3158 mb_return = wctomb(mbc, c); 3159 3160 if(mb_return == -1) 3161 return WEOF; 3162 3163 /* Output all characters */ 3164 if (fwrite(mbc, mb_return, 1, stream) != 1) 3165 return WEOF; 3166 } 3167 else 3168 { 3169 if (fwrite(&c, sizeof(c), 1, stream) != 1) 3170 return WEOF; 3171 } 3172 3173 return c; 3174 } 3175 3176 /********************************************************************* 3177 * _fputwchar (MSVCRT.@) 3178 */ 3179 wint_t CDECL _fputwchar(wint_t wc) 3180 { 3181 return fputwc(wc, stdout); 3182 } 3183 3184 /********************************************************************* 3185 * _wfsopen (MSVCRT.@) 3186 */ 3187 FILE * CDECL _wfsopen(const wchar_t *path, const wchar_t *mode, int share) 3188 { 3189 FILE* file; 3190 int open_flags, stream_flags, fd; 3191 3192 TRACE("(%s,%s)\n", debugstr_w(path), debugstr_w(mode)); 3193 3194 /* map mode string to open() flags. "man fopen" for possibilities. */ 3195 if (msvcrt_get_flags(mode, &open_flags, &stream_flags) == -1) 3196 return NULL; 3197 3198 LOCK_FILES(); 3199 fd = _wsopen(path, open_flags, share, _S_IREAD | _S_IWRITE); 3200 if (fd < 0) 3201 file = NULL; 3202 else if ((file = msvcrt_alloc_fp()) && msvcrt_init_fp(file, fd, stream_flags) 3203 != -1) 3204 TRACE(":fd (%d) mode (%s) FILE* (%p)\n", fd, debugstr_w(mode), file); 3205 else if (file) 3206 { 3207 file->_flag = 0; 3208 file = NULL; 3209 } 3210 3211 TRACE(":got (%p)\n",file); 3212 if (fd >= 0 && !file) 3213 _close(fd); 3214 UNLOCK_FILES(); 3215 return file; 3216 } 3217 3218 /********************************************************************* 3219 * _fsopen (MSVCRT.@) 3220 */ 3221 FILE * CDECL _fsopen(const char *path, const char *mode, int share) 3222 { 3223 FILE *ret; 3224 wchar_t *pathW = NULL, *modeW = NULL; 3225 3226 if (path && !(pathW = msvcrt_wstrdupa(path))) { 3227 _invalid_parameter(NULL, NULL, NULL, 0, 0); 3228 *_errno() = EINVAL; 3229 return NULL; 3230 } 3231 if (mode && !(modeW = msvcrt_wstrdupa(mode))) 3232 { 3233 free(pathW); 3234 _invalid_parameter(NULL, NULL, NULL, 0, 0); 3235 *_errno() = EINVAL; 3236 return NULL; 3237 } 3238 3239 ret = _wfsopen(pathW, modeW, share); 3240 3241 free(pathW); 3242 free(modeW); 3243 return ret; 3244 } 3245 3246 /********************************************************************* 3247 * fopen (MSVCRT.@) 3248 */ 3249 FILE * CDECL fopen(const char *path, const char *mode) 3250 { 3251 return _fsopen( path, mode, _SH_DENYNO ); 3252 } 3253 3254 /********************************************************************* 3255 * fopen_s (MSVCRT.@) 3256 */ 3257 int CDECL fopen_s(FILE** pFile, 3258 const char *filename, const char *mode) 3259 { 3260 if (!MSVCRT_CHECK_PMT(pFile != NULL)) return EINVAL; 3261 if (!MSVCRT_CHECK_PMT(filename != NULL)) return EINVAL; 3262 if (!MSVCRT_CHECK_PMT(mode != NULL)) return EINVAL; 3263 3264 *pFile = fopen(filename, mode); 3265 3266 if(!*pFile) 3267 return *_errno(); 3268 return 0; 3269 } 3270 3271 /********************************************************************* 3272 * _wfopen (MSVCRT.@) 3273 */ 3274 FILE * CDECL _wfopen(const wchar_t *path, const wchar_t *mode) 3275 { 3276 return _wfsopen( path, mode, _SH_DENYNO ); 3277 } 3278 3279 /********************************************************************* 3280 * _wfopen_s (MSVCRT.@) 3281 */ 3282 int CDECL _wfopen_s(FILE** pFile, const wchar_t *filename, 3283 const wchar_t *mode) 3284 { 3285 if (!MSVCRT_CHECK_PMT(pFile != NULL) || !MSVCRT_CHECK_PMT(filename != NULL) || 3286 !MSVCRT_CHECK_PMT(mode != NULL)) { 3287 *_errno() = EINVAL; 3288 return EINVAL; 3289 } 3290 3291 *pFile = _wfopen(filename, mode); 3292 3293 if(!*pFile) 3294 return *_errno(); 3295 return 0; 3296 } 3297 3298 /* fputc calls _flsbuf which calls fputc */ 3299 int CDECL _flsbuf(int c, FILE* file); 3300 3301 /********************************************************************* 3302 * fputc (MSVCRT.@) 3303 */ 3304 int CDECL fputc(int c, FILE* file) 3305 { 3306 int res; 3307 3308 _lock_file(file); 3309 if(file->_cnt>0) { 3310 *file->_ptr++=c; 3311 file->_cnt--; 3312 if (c == '\n') 3313 { 3314 res = msvcrt_flush_buffer(file); 3315 _unlock_file(file); 3316 return res ? res : c; 3317 } 3318 else { 3319 _unlock_file(file); 3320 return c & 0xff; 3321 } 3322 } else { 3323 res = _flsbuf(c, file); 3324 _unlock_file(file); 3325 return res; 3326 } 3327 } 3328 3329 /********************************************************************* 3330 * _fputchar (MSVCRT.@) 3331 */ 3332 int CDECL _fputchar(int c) 3333 { 3334 return fputc(c, stdout); 3335 } 3336 3337 /********************************************************************* 3338 * fread (MSVCRT.@) 3339 */ 3340 size_t CDECL fread(void *ptr, size_t size, size_t nmemb, FILE* file) 3341 { 3342 size_t rcnt=size * nmemb; 3343 size_t read=0; 3344 size_t pread=0; 3345 3346 if(!rcnt) 3347 return 0; 3348 3349 _lock_file(file); 3350 3351 /* first buffered data */ 3352 if(file->_cnt>0) { 3353 int pcnt= (rcnt>file->_cnt)? file->_cnt:rcnt; 3354 memcpy(ptr, file->_ptr, pcnt); 3355 file->_cnt -= pcnt; 3356 file->_ptr += pcnt; 3357 read += pcnt ; 3358 rcnt -= pcnt ; 3359 ptr = (char*)ptr + pcnt; 3360 } else if(!(file->_flag & _IOREAD )) { 3361 if(file->_flag & _IORW) { 3362 file->_flag |= _IOREAD; 3363 } else { 3364 _unlock_file(file); 3365 return 0; 3366 } 3367 } 3368 3369 if(rcnt>0 && !(file->_flag & (_IONBF | _IOMYBUF | _USERBUF))) 3370 msvcrt_alloc_buffer(file); 3371 3372 while(rcnt>0) 3373 { 3374 int i; 3375 if (!file->_cnt && rcnt<BUFSIZ && (file->_flag & (_IOMYBUF | _USERBUF))) { 3376 file->_cnt = _read(file->_file, file->_base, file->_bufsiz); 3377 file->_ptr = file->_base; 3378 i = (file->_cnt<rcnt) ? file->_cnt : rcnt; 3379 /* If the buffer fill reaches eof but fread wouldn't, clear eof. */ 3380 if (i > 0 && i < file->_cnt) { 3381 get_ioinfo_nolock(file->_file)->wxflag &= ~WX_ATEOF; 3382 file->_flag &= ~_IOEOF; 3383 } 3384 if (i > 0) { 3385 memcpy(ptr, file->_ptr, i); 3386 file->_cnt -= i; 3387 file->_ptr += i; 3388 } 3389 } else if (rcnt > INT_MAX) { 3390 i = _read(file->_file, ptr, INT_MAX); 3391 } else if (rcnt < BUFSIZ) { 3392 i = _read(file->_file, ptr, rcnt); 3393 } else { 3394 i = _read(file->_file, ptr, rcnt - BUFSIZ/2); 3395 } 3396 pread += i; 3397 rcnt -= i; 3398 ptr = (char *)ptr+i; 3399 /* expose feof condition in the flags 3400 * MFC tests file->_flag for feof, and doesn't call feof()) 3401 */ 3402 if (get_ioinfo_nolock(file->_file)->wxflag & WX_ATEOF) 3403 file->_flag |= _IOEOF; 3404 else if (i == -1) 3405 { 3406 file->_flag |= _IOERR; 3407 pread = 0; 3408 rcnt = 0; 3409 } 3410 if (i < 1) break; 3411 } 3412 read+=pread; 3413 _unlock_file(file); 3414 return read / size; 3415 } 3416 3417 /********************************************************************* 3418 * _wfreopen (MSVCRT.@) 3419 * 3420 */ 3421 FILE* CDECL _wfreopen(const wchar_t *path, const wchar_t *mode, FILE* file) 3422 { 3423 int open_flags, stream_flags, fd; 3424 3425 TRACE(":path (%s) mode (%s) file (%p) fd (%d)\n", debugstr_w(path), debugstr_w(mode), file, file ? file->_file : -1); 3426 3427 LOCK_FILES(); 3428 if (!file || ((fd = file->_file) < 0)) 3429 file = NULL; 3430 else 3431 { 3432 fclose(file); 3433 /* map mode string to open() flags. "man fopen" for possibilities. */ 3434 if (msvcrt_get_flags(mode, &open_flags, &stream_flags) == -1) 3435 file = NULL; 3436 else 3437 { 3438 fd = _wopen(path, open_flags, _S_IREAD | _S_IWRITE); 3439 if (fd < 0) 3440 file = NULL; 3441 else if (msvcrt_init_fp(file, fd, stream_flags) == -1) 3442 { 3443 file->_flag = 0; 3444 WARN(":failed-last error (%d)\n",GetLastError()); 3445 _dosmaperr(GetLastError()); 3446 file = NULL; 3447 } 3448 } 3449 } 3450 UNLOCK_FILES(); 3451 return file; 3452 } 3453 3454 /********************************************************************* 3455 * freopen (MSVCRT.@) 3456 * 3457 */ 3458 FILE* CDECL freopen(const char *path, const char *mode, FILE* file) 3459 { 3460 FILE *ret; 3461 wchar_t *pathW = NULL, *modeW = NULL; 3462 3463 if (path && !(pathW = msvcrt_wstrdupa(path))) return NULL; 3464 if (mode && !(modeW = msvcrt_wstrdupa(mode))) 3465 { 3466 free(pathW); 3467 return NULL; 3468 } 3469 3470 ret = _wfreopen(pathW, modeW, file); 3471 3472 free(pathW); 3473 free(modeW); 3474 return ret; 3475 } 3476 3477 /********************************************************************* 3478 * fsetpos (MSVCRT.@) 3479 */ 3480 int CDECL fsetpos(FILE* file, const fpos_t *pos) 3481 { 3482 int ret; 3483 3484 _lock_file(file); 3485 /* Note that all this has been lifted 'as is' from fseek */ 3486 if(file->_flag & _IOWRT) 3487 msvcrt_flush_buffer(file); 3488 3489 /* Discard buffered input */ 3490 file->_cnt = 0; 3491 file->_ptr = file->_base; 3492 3493 /* Reset direction of i/o */ 3494 if(file->_flag & _IORW) { 3495 file->_flag &= ~(_IOREAD|_IOWRT); 3496 } 3497 3498 ret = (_lseeki64(file->_file,*pos,SEEK_SET) == -1) ? -1 : 0; 3499 _unlock_file(file); 3500 return ret; 3501 } 3502 3503 /********************************************************************* 3504 * _ftelli64 (MSVCRT.@) 3505 */ 3506 __int64 CDECL _ftelli64(FILE* file) 3507 { 3508 __int64 pos; 3509 3510 _lock_file(file); 3511 pos = _telli64(file->_file); 3512 if(pos == -1) { 3513 _unlock_file(file); 3514 return -1; 3515 } 3516 if(file->_flag & (_IOMYBUF | _USERBUF)) { 3517 if(file->_flag & _IOWRT) { 3518 pos += file->_ptr - file->_base; 3519 3520 if(get_ioinfo_nolock(file->_file)->wxflag & WX_TEXT) { 3521 char *p; 3522 3523 for(p=file->_base; p<file->_ptr; p++) 3524 if(*p == '\n') 3525 pos++; 3526 } 3527 } else if(!file->_cnt) { /* nothing to do */ 3528 } else if(_lseeki64(file->_file, 0, SEEK_END)==pos) { 3529 int i; 3530 3531 pos -= file->_cnt; 3532 if(get_ioinfo_nolock(file->_file)->wxflag & WX_TEXT) { 3533 for(i=0; i<file->_cnt; i++) 3534 if(file->_ptr[i] == '\n') 3535 pos--; 3536 } 3537 } else { 3538 char *p; 3539 3540 if(_lseeki64(file->_file, pos, SEEK_SET) != pos) { 3541 _unlock_file(file); 3542 return -1; 3543 } 3544 3545 pos -= file->_bufsiz; 3546 pos += file->_ptr - file->_base; 3547 3548 if(get_ioinfo_nolock(file->_file)->wxflag & WX_TEXT) { 3549 if(get_ioinfo_nolock(file->_file)->wxflag & WX_READNL) 3550 pos--; 3551 3552 for(p=file->_base; p<file->_ptr; p++) 3553 if(*p == '\n') 3554 pos++; 3555 } 3556 } 3557 } 3558 3559 _unlock_file(file); 3560 return pos; 3561 } 3562 3563 /********************************************************************* 3564 * ftell (MSVCRT.@) 3565 */ 3566 LONG CDECL ftell(FILE* file) 3567 { 3568 return (LONG)_ftelli64(file); 3569 } 3570 3571 /********************************************************************* 3572 * fgetpos (MSVCRT.@) 3573 */ 3574 int CDECL fgetpos(FILE* file, fpos_t *pos) 3575 { 3576 *pos = _ftelli64(file); 3577 if(*pos == -1) 3578 return -1; 3579 return 0; 3580 } 3581 3582 /********************************************************************* 3583 * fputs (MSVCRT.@) 3584 */ 3585 int CDECL fputs(const char *s, FILE* file) 3586 { 3587 size_t len = strlen(s); 3588 int ret; 3589 3590 _lock_file(file); 3591 ret = fwrite(s, sizeof(*s), len, file) == len ? 0 : EOF; 3592 _unlock_file(file); 3593 return ret; 3594 } 3595 3596 /********************************************************************* 3597 * fputws (MSVCRT.@) 3598 */ 3599 int CDECL fputws(const wchar_t *s, FILE* file) 3600 { 3601 size_t i, len = strlenW(s); 3602 BOOL tmp_buf; 3603 int ret; 3604 3605 _lock_file(file); 3606 if (!(get_ioinfo_nolock(file->_file)->wxflag & WX_TEXT)) { 3607 ret = fwrite(s,sizeof(*s),len,file) == len ? 0 : EOF; 3608 _unlock_file(file); 3609 return ret; 3610 } 3611 3612 tmp_buf = add_std_buffer(file); 3613 for (i=0; i<len; i++) { 3614 if(fputwc(s[i], file) == WEOF) { 3615 if(tmp_buf) remove_std_buffer(file); 3616 _unlock_file(file); 3617 return WEOF; 3618 } 3619 } 3620 3621 if(tmp_buf) remove_std_buffer(file); 3622 _unlock_file(file); 3623 return 0; 3624 } 3625 3626 /********************************************************************* 3627 * getchar (MSVCRT.@) 3628 */ 3629 int CDECL getchar(void) 3630 { 3631 return fgetc(stdin); 3632 } 3633 3634 /********************************************************************* 3635 * getc (MSVCRT.@) 3636 */ 3637 int CDECL getc(FILE* file) 3638 { 3639 return fgetc(file); 3640 } 3641 3642 /********************************************************************* 3643 * gets (MSVCRT.@) 3644 */ 3645 char * CDECL gets(char *buf) 3646 { 3647 int cc; 3648 char * buf_start = buf; 3649 3650 _lock_file(stdin); 3651 for(cc = fgetc(stdin); cc != EOF && cc != '\n'; 3652 cc = fgetc(stdin)) 3653 if(cc != '\r') *buf++ = (char)cc; 3654 3655 *buf = '\0'; 3656 3657 TRACE("got '%s'\n", buf_start); 3658 _unlock_file(stdin); 3659 return buf_start; 3660 } 3661 3662 /********************************************************************* 3663 * _getws (MSVCRT.@) 3664 */ 3665 wchar_t* CDECL _getws(wchar_t* buf) 3666 { 3667 wint_t cc; 3668 wchar_t* ws = buf; 3669 3670 _lock_file(stdin); 3671 for (cc = fgetwc(stdin); cc != WEOF && cc != '\n'; 3672 cc = fgetwc(stdin)) 3673 { 3674 if (cc != '\r') 3675 *buf++ = (wchar_t)cc; 3676 } 3677 *buf = '\0'; 3678 3679 TRACE("got %s\n", debugstr_w(ws)); 3680 _unlock_file(stdin); 3681 return ws; 3682 } 3683 3684 /********************************************************************* 3685 * putc (MSVCRT.@) 3686 */ 3687 int CDECL putc(int c, FILE* file) 3688 { 3689 return fputc(c, file); 3690 } 3691 3692 /********************************************************************* 3693 * putchar (MSVCRT.@) 3694 */ 3695 int CDECL putchar(int c) 3696 { 3697 return fputc(c, stdout); 3698 } 3699 3700 /********************************************************************* 3701 * _putwch (MSVCRT.@) 3702 */ 3703 wint_t CDECL _putwch(wchar_t c) 3704 { 3705 return fputwc(c, stdout); 3706 } 3707 3708 /********************************************************************* 3709 * puts (MSVCRT.@) 3710 */ 3711 int CDECL puts(const char *s) 3712 { 3713 size_t len = strlen(s); 3714 int ret; 3715 3716 _lock_file(stdout); 3717 if(fwrite(s, sizeof(*s), len, stdout) != len) { 3718 _unlock_file(stdout); 3719 return EOF; 3720 } 3721 3722 ret = fwrite("\n",1,1,stdout) == 1 ? 0 : EOF; 3723 _unlock_file(stdout); 3724 return ret; 3725 } 3726 3727 /********************************************************************* 3728 * _putws (MSVCRT.@) 3729 */ 3730 int CDECL _putws(const wchar_t *s) 3731 { 3732 static const wchar_t nl = '\n'; 3733 size_t len = strlenW(s); 3734 int ret; 3735 3736 _lock_file(stdout); 3737 if(fwrite(s, sizeof(*s), len, stdout) != len) { 3738 _unlock_file(stdout); 3739 return EOF; 3740 } 3741 3742 ret = fwrite(&nl,sizeof(nl),1,stdout) == 1 ? 0 : EOF; 3743 _unlock_file(stdout); 3744 return ret; 3745 } 3746 3747 /********************************************************************* 3748 * remove (MSVCRT.@) 3749 */ 3750 int CDECL remove(const char *path) 3751 { 3752 TRACE("(%s)\n",path); 3753 if (DeleteFileA(path)) 3754 return 0; 3755 TRACE(":failed (%d)\n",GetLastError()); 3756 _dosmaperr(GetLastError()); 3757 return -1; 3758 } 3759 3760 /********************************************************************* 3761 * _wremove (MSVCRT.@) 3762 */ 3763 int CDECL _wremove(const wchar_t *path) 3764 { 3765 TRACE("(%s)\n",debugstr_w(path)); 3766 if (DeleteFileW(path)) 3767 return 0; 3768 TRACE(":failed (%d)\n",GetLastError()); 3769 _dosmaperr(GetLastError()); 3770 return -1; 3771 } 3772 3773 /********************************************************************* 3774 * rename (MSVCRT.@) 3775 */ 3776 int CDECL rename(const char *oldpath,const char *newpath) 3777 { 3778 TRACE(":from %s to %s\n",oldpath,newpath); 3779 if (MoveFileExA(oldpath, newpath, MOVEFILE_COPY_ALLOWED)) 3780 return 0; 3781 TRACE(":failed (%d)\n",GetLastError()); 3782 _dosmaperr(GetLastError()); 3783 return -1; 3784 } 3785 3786 /********************************************************************* 3787 * _wrename (MSVCRT.@) 3788 */ 3789 int CDECL _wrename(const wchar_t *oldpath,const wchar_t *newpath) 3790 { 3791 TRACE(":from %s to %s\n",debugstr_w(oldpath),debugstr_w(newpath)); 3792 if (MoveFileExW(oldpath, newpath, MOVEFILE_COPY_ALLOWED)) 3793 return 0; 3794 TRACE(":failed (%d)\n",GetLastError()); 3795 _dosmaperr(GetLastError()); 3796 return -1; 3797 } 3798 3799 /********************************************************************* 3800 * setvbuf (MSVCRT.@) 3801 */ 3802 int CDECL setvbuf(FILE* file, char *buf, int mode, size_t size) 3803 { 3804 if(!MSVCRT_CHECK_PMT(file != NULL)) return -1; 3805 if(!MSVCRT_CHECK_PMT(mode==_IONBF || mode==_IOFBF || mode==_IOLBF)) return -1; 3806 if(!MSVCRT_CHECK_PMT(mode==_IONBF || (size>=2 && size<=INT_MAX))) return -1; 3807 3808 _lock_file(file); 3809 3810 fflush(file); 3811 if(file->_flag & _IOMYBUF) 3812 free(file->_base); 3813 file->_flag &= ~(_IONBF | _IOMYBUF | _USERBUF); 3814 file->_cnt = 0; 3815 3816 if(mode == _IONBF) { 3817 file->_flag |= _IONBF; 3818 file->_base = file->_ptr = (char*)&file->_charbuf; 3819 file->_bufsiz = 2; 3820 }else if(buf) { 3821 file->_base = file->_ptr = buf; 3822 file->_flag |= _USERBUF; 3823 file->_bufsiz = size; 3824 }else { 3825 file->_base = file->_ptr = malloc(size); 3826 if(!file->_base) { 3827 file->_bufsiz = 0; 3828 _unlock_file(file); 3829 return -1; 3830 } 3831 3832 file->_flag |= _IOMYBUF; 3833 file->_bufsiz = size; 3834 } 3835 _unlock_file(file); 3836 return 0; 3837 } 3838 3839 /********************************************************************* 3840 * setbuf (MSVCRT.@) 3841 */ 3842 void CDECL setbuf(FILE* file, char *buf) 3843 { 3844 setvbuf(file, buf, buf ? _IOFBF : _IONBF, BUFSIZ); 3845 } 3846 3847 /********************************************************************* 3848 * tmpnam (MSVCRT.@) 3849 */ 3850 char * CDECL tmpnam(char *s) 3851 { 3852 char tmpstr[16]; 3853 char *p; 3854 int count, size; 3855 3856 if (!s) { 3857 thread_data_t *data = msvcrt_get_thread_data(); 3858 3859 if(!data->tmpnam_buffer) 3860 data->tmpnam_buffer = malloc(MAX_PATH); 3861 3862 s = data->tmpnam_buffer; 3863 } 3864 3865 msvcrt_int_to_base32(GetCurrentProcessId(), tmpstr); 3866 p = s + sprintf(s, "\\s%s.", tmpstr); 3867 for (count = 0; count < TMP_MAX; count++) 3868 { 3869 size = msvcrt_int_to_base32(tmpnam_unique++, tmpstr); 3870 memcpy(p, tmpstr, size); 3871 p[size] = '\0'; 3872 if (GetFileAttributesA(s) == INVALID_FILE_ATTRIBUTES && 3873 GetLastError() == ERROR_FILE_NOT_FOUND) 3874 break; 3875 } 3876 return s; 3877 } 3878 3879 /********************************************************************* 3880 * _wtmpnam (MSVCRT.@) 3881 */ 3882 wchar_t * CDECL _wtmpnam(wchar_t *s) 3883 { 3884 static const wchar_t format[] = {'\\','s','%','s','.',0}; 3885 wchar_t tmpstr[16]; 3886 wchar_t *p; 3887 int count, size; 3888 if (!s) { 3889 thread_data_t *data = msvcrt_get_thread_data(); 3890 3891 if(!data->wtmpnam_buffer) 3892 data->wtmpnam_buffer = malloc(sizeof(wchar_t[MAX_PATH])); 3893 3894 s = data->wtmpnam_buffer; 3895 } 3896 3897 msvcrt_int_to_base32_w(GetCurrentProcessId(), tmpstr); 3898 p = s + _snwprintf(s, MAX_PATH, format, tmpstr); 3899 for (count = 0; count < TMP_MAX; count++) 3900 { 3901 size = msvcrt_int_to_base32_w(tmpnam_unique++, tmpstr); 3902 memcpy(p, tmpstr, size*sizeof(wchar_t)); 3903 p[size] = '\0'; 3904 if (GetFileAttributesW(s) == INVALID_FILE_ATTRIBUTES && 3905 GetLastError() == ERROR_FILE_NOT_FOUND) 3906 break; 3907 } 3908 return s; 3909 } 3910 3911 /********************************************************************* 3912 * tmpfile (MSVCRT.@) 3913 */ 3914 FILE* CDECL tmpfile(void) 3915 { 3916 char *filename = tmpnam(NULL); 3917 int fd; 3918 FILE* file = NULL; 3919 3920 LOCK_FILES(); 3921 fd = _open(filename, _O_CREAT | _O_BINARY | _O_RDWR | _O_TEMPORARY, 3922 _S_IREAD | _S_IWRITE); 3923 if (fd != -1 && (file = msvcrt_alloc_fp())) 3924 { 3925 if (msvcrt_init_fp(file, fd, _IORW) == -1) 3926 { 3927 file->_flag = 0; 3928 file = NULL; 3929 } 3930 else file->_tmpfname = _strdup(filename); 3931 } 3932 3933 if(fd != -1 && !file) 3934 _close(fd); 3935 UNLOCK_FILES(); 3936 return file; 3937 } 3938 3939 /********************************************************************* 3940 * ungetc (MSVCRT.@) 3941 */ 3942 int CDECL ungetc(int c, FILE * file) 3943 { 3944 if(!MSVCRT_CHECK_PMT(file != NULL)) return EOF; 3945 3946 if (c == EOF || !(file->_flag&_IOREAD || 3947 (file->_flag&_IORW && !(file->_flag&_IOWRT)))) 3948 return EOF; 3949 3950 _lock_file(file); 3951 if((!(file->_flag & (_IONBF | _IOMYBUF | _USERBUF)) 3952 && msvcrt_alloc_buffer(file)) 3953 || (!file->_cnt && file->_ptr==file->_base)) 3954 file->_ptr++; 3955 3956 if(file->_ptr>file->_base) { 3957 file->_ptr--; 3958 if(file->_flag & _IOSTRG) { 3959 if(*file->_ptr != c) { 3960 file->_ptr++; 3961 _unlock_file(file); 3962 return EOF; 3963 } 3964 }else { 3965 *file->_ptr = c; 3966 } 3967 file->_cnt++; 3968 clearerr(file); 3969 file->_flag |= _IOREAD; 3970 _unlock_file(file); 3971 return c; 3972 } 3973 3974 _unlock_file(file); 3975 return EOF; 3976 } 3977 3978 /********************************************************************* 3979 * ungetwc (MSVCRT.@) 3980 */ 3981 wint_t CDECL ungetwc(wint_t wc, FILE * file) 3982 { 3983 wchar_t mwc = wc; 3984 3985 if (wc == WEOF) 3986 return WEOF; 3987 3988 _lock_file(file); 3989 3990 if((get_ioinfo_nolock(file->_file)->exflag & (EF_UTF8 | EF_UTF16)) 3991 || !(get_ioinfo_nolock(file->_file)->wxflag & WX_TEXT)) { 3992 unsigned char * pp = (unsigned char *)&mwc; 3993 int i; 3994 3995 for(i=sizeof(wchar_t)-1;i>=0;i--) { 3996 if(pp[i] != ungetc(pp[i],file)) { 3997 _unlock_file(file); 3998 return WEOF; 3999 } 4000 } 4001 }else { 4002 char mbs[MB_LEN_MAX]; 4003 int len; 4004 4005 len = wctomb(mbs, mwc); 4006 if(len == -1) { 4007 _unlock_file(file); 4008 return WEOF; 4009 } 4010 4011 for(len--; len>=0; len--) { 4012 if(mbs[len] != ungetc(mbs[len], file)) { 4013 _unlock_file(file); 4014 return WEOF; 4015 } 4016 } 4017 } 4018 4019 _unlock_file(file); 4020 return mwc; 4021 } 4022 4023 4024 4025 /********************************************************************* 4026 * _getmaxstdio (MSVCRT.@) 4027 */ 4028 int CDECL _getmaxstdio(void) 4029 { 4030 return MSVCRT_max_streams; 4031 } 4032 4033 /********************************************************************* 4034 * _setmaxstdio (MSVCRT.@) 4035 */ 4036 int CDECL _setmaxstdio(int newmax) 4037 { 4038 TRACE("%d\n", newmax); 4039 4040 if(newmax<_IOB_ENTRIES || newmax>MSVCRT_MAX_FILES || newmax<MSVCRT_stream_idx) 4041 return -1; 4042 4043 MSVCRT_max_streams = newmax; 4044 return MSVCRT_max_streams; 4045 } 4046