Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 1 | /* |
| 2 | * File handling functions |
| 3 | * |
| 4 | * Copyright 1993 John Burton |
Alexandre Julliard | b592cbb | 2004-05-01 02:44:00 +0000 | [diff] [blame] | 5 | * Copyright 1996, 2004 Alexandre Julliard |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 6 | * |
| 7 | * This library is free software; you can redistribute it and/or |
| 8 | * modify it under the terms of the GNU Lesser General Public |
| 9 | * License as published by the Free Software Foundation; either |
| 10 | * version 2.1 of the License, or (at your option) any later version. |
| 11 | * |
| 12 | * This library is distributed in the hope that it will be useful, |
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 15 | * Lesser General Public License for more details. |
| 16 | * |
| 17 | * You should have received a copy of the GNU Lesser General Public |
| 18 | * License along with this library; if not, write to the Free Software |
| 19 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 20 | */ |
| 21 | |
| 22 | #include "config.h" |
| 23 | #include "wine/port.h" |
| 24 | |
Alexandre Julliard | e37c6e1 | 2003-09-05 23:08:26 +0000 | [diff] [blame] | 25 | #include <stdarg.h> |
Hans Leidekker | 84e9eb6 | 2004-10-11 19:53:13 +0000 | [diff] [blame] | 26 | #include <stdio.h> |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 27 | #include <errno.h> |
Alexandre Julliard | e37c6e1 | 2003-09-05 23:08:26 +0000 | [diff] [blame] | 28 | |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 29 | #define NONAMELESSUNION |
| 30 | #define NONAMELESSSTRUCT |
| 31 | #include "winerror.h" |
Alexandre Julliard | e37c6e1 | 2003-09-05 23:08:26 +0000 | [diff] [blame] | 32 | #include "ntstatus.h" |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 33 | #include "windef.h" |
| 34 | #include "winbase.h" |
Alexandre Julliard | e37c6e1 | 2003-09-05 23:08:26 +0000 | [diff] [blame] | 35 | #include "winreg.h" |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 36 | #include "winternl.h" |
Alexandre Julliard | b592cbb | 2004-05-01 02:44:00 +0000 | [diff] [blame] | 37 | #include "winioctl.h" |
Eric Pouech | 9bd4f6b | 2003-06-26 02:08:17 +0000 | [diff] [blame] | 38 | #include "wincon.h" |
Dimitrie O. Paun | 2eaefbd | 2003-12-08 21:58:55 +0000 | [diff] [blame] | 39 | #include "wine/winbase16.h" |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 40 | #include "kernel_private.h" |
| 41 | |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 42 | #include "wine/exception.h" |
| 43 | #include "excpt.h" |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 44 | #include "wine/unicode.h" |
| 45 | #include "wine/debug.h" |
Eric Pouech | 9bd4f6b | 2003-06-26 02:08:17 +0000 | [diff] [blame] | 46 | #include "async.h" |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 47 | |
| 48 | WINE_DEFAULT_DEBUG_CHANNEL(file); |
| 49 | |
Eric Pouech | 768008f | 2004-01-14 04:34:20 +0000 | [diff] [blame] | 50 | HANDLE dos_handles[DOS_TABLE_SIZE]; |
| 51 | |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 52 | /* info structure for FindFirstFile handle */ |
| 53 | typedef struct |
| 54 | { |
Alexandre Julliard | f435914 | 2004-06-15 00:55:04 +0000 | [diff] [blame] | 55 | DWORD magic; /* magic number */ |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 56 | HANDLE handle; /* handle to directory */ |
| 57 | CRITICAL_SECTION cs; /* crit section protecting this structure */ |
| 58 | UNICODE_STRING mask; /* file mask */ |
| 59 | BOOL is_root; /* is directory the root of the drive? */ |
| 60 | UINT data_pos; /* current position in dir data */ |
| 61 | UINT data_len; /* length of dir data */ |
| 62 | BYTE data[8192]; /* directory data */ |
| 63 | } FIND_FIRST_INFO; |
| 64 | |
Alexandre Julliard | f435914 | 2004-06-15 00:55:04 +0000 | [diff] [blame] | 65 | #define FIND_FIRST_MAGIC 0xc0ffee11 |
| 66 | |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 67 | static BOOL oem_file_apis; |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 68 | |
| 69 | static WINE_EXCEPTION_FILTER(page_fault) |
| 70 | { |
| 71 | if (GetExceptionCode() == EXCEPTION_ACCESS_VIOLATION) |
| 72 | return EXCEPTION_EXECUTE_HANDLER; |
| 73 | return EXCEPTION_CONTINUE_SEARCH; |
| 74 | } |
| 75 | |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 76 | |
| 77 | /*********************************************************************** |
| 78 | * create_file_OF |
| 79 | * |
| 80 | * Wrapper for CreateFile that takes OF_* mode flags. |
| 81 | */ |
Alexandre Julliard | 2f8fd10 | 2004-05-03 20:20:54 +0000 | [diff] [blame] | 82 | static HANDLE create_file_OF( LPCSTR path, INT mode ) |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 83 | { |
Alexandre Julliard | 2f8fd10 | 2004-05-03 20:20:54 +0000 | [diff] [blame] | 84 | DWORD access, sharing, creation; |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 85 | |
Alexandre Julliard | 2f8fd10 | 2004-05-03 20:20:54 +0000 | [diff] [blame] | 86 | if (mode & OF_CREATE) |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 87 | { |
Alexandre Julliard | 2f8fd10 | 2004-05-03 20:20:54 +0000 | [diff] [blame] | 88 | creation = CREATE_ALWAYS; |
| 89 | access = GENERIC_READ | GENERIC_WRITE; |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 90 | } |
Alexandre Julliard | 2f8fd10 | 2004-05-03 20:20:54 +0000 | [diff] [blame] | 91 | else |
| 92 | { |
| 93 | creation = OPEN_EXISTING; |
| 94 | switch(mode & 0x03) |
| 95 | { |
| 96 | case OF_READ: access = GENERIC_READ; break; |
| 97 | case OF_WRITE: access = GENERIC_WRITE; break; |
| 98 | case OF_READWRITE: access = GENERIC_READ | GENERIC_WRITE; break; |
| 99 | default: access = 0; break; |
| 100 | } |
| 101 | } |
| 102 | |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 103 | switch(mode & 0x70) |
| 104 | { |
| 105 | case OF_SHARE_EXCLUSIVE: sharing = 0; break; |
| 106 | case OF_SHARE_DENY_WRITE: sharing = FILE_SHARE_READ; break; |
| 107 | case OF_SHARE_DENY_READ: sharing = FILE_SHARE_WRITE; break; |
| 108 | case OF_SHARE_DENY_NONE: |
| 109 | case OF_SHARE_COMPAT: |
| 110 | default: sharing = FILE_SHARE_READ | FILE_SHARE_WRITE; break; |
| 111 | } |
| 112 | return CreateFileA( path, access, sharing, NULL, creation, FILE_ATTRIBUTE_NORMAL, 0 ); |
| 113 | } |
| 114 | |
| 115 | |
| 116 | /*********************************************************************** |
| 117 | * FILE_SetDosError |
| 118 | * |
| 119 | * Set the DOS error code from errno. |
| 120 | */ |
| 121 | void FILE_SetDosError(void) |
| 122 | { |
| 123 | int save_errno = errno; /* errno gets overwritten by printf */ |
| 124 | |
| 125 | TRACE("errno = %d %s\n", errno, strerror(errno)); |
| 126 | switch (save_errno) |
| 127 | { |
| 128 | case EAGAIN: |
| 129 | SetLastError( ERROR_SHARING_VIOLATION ); |
| 130 | break; |
| 131 | case EBADF: |
| 132 | SetLastError( ERROR_INVALID_HANDLE ); |
| 133 | break; |
| 134 | case ENOSPC: |
| 135 | SetLastError( ERROR_HANDLE_DISK_FULL ); |
| 136 | break; |
| 137 | case EACCES: |
| 138 | case EPERM: |
| 139 | case EROFS: |
| 140 | SetLastError( ERROR_ACCESS_DENIED ); |
| 141 | break; |
| 142 | case EBUSY: |
| 143 | SetLastError( ERROR_LOCK_VIOLATION ); |
| 144 | break; |
| 145 | case ENOENT: |
| 146 | SetLastError( ERROR_FILE_NOT_FOUND ); |
| 147 | break; |
| 148 | case EISDIR: |
| 149 | SetLastError( ERROR_CANNOT_MAKE ); |
| 150 | break; |
| 151 | case ENFILE: |
| 152 | case EMFILE: |
| 153 | SetLastError( ERROR_TOO_MANY_OPEN_FILES ); |
| 154 | break; |
| 155 | case EEXIST: |
| 156 | SetLastError( ERROR_FILE_EXISTS ); |
| 157 | break; |
| 158 | case EINVAL: |
| 159 | case ESPIPE: |
| 160 | SetLastError( ERROR_SEEK ); |
| 161 | break; |
| 162 | case ENOTEMPTY: |
| 163 | SetLastError( ERROR_DIR_NOT_EMPTY ); |
| 164 | break; |
| 165 | case ENOEXEC: |
| 166 | SetLastError( ERROR_BAD_FORMAT ); |
| 167 | break; |
| 168 | case ENOTDIR: |
| 169 | SetLastError( ERROR_PATH_NOT_FOUND ); |
| 170 | break; |
| 171 | case EXDEV: |
| 172 | SetLastError( ERROR_NOT_SAME_DEVICE ); |
| 173 | break; |
| 174 | default: |
| 175 | WARN("unknown file error: %s\n", strerror(save_errno) ); |
| 176 | SetLastError( ERROR_GEN_FAILURE ); |
| 177 | break; |
| 178 | } |
| 179 | errno = save_errno; |
| 180 | } |
| 181 | |
| 182 | |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 183 | /*********************************************************************** |
| 184 | * FILE_name_AtoW |
| 185 | * |
| 186 | * Convert a file name to Unicode, taking into account the OEM/Ansi API mode. |
| 187 | * |
| 188 | * If alloc is FALSE uses the TEB static buffer, so it can only be used when |
| 189 | * there is no possibility for the function to do that twice, taking into |
| 190 | * account any called function. |
| 191 | */ |
| 192 | WCHAR *FILE_name_AtoW( LPCSTR name, BOOL alloc ) |
| 193 | { |
| 194 | ANSI_STRING str; |
| 195 | UNICODE_STRING strW, *pstrW; |
| 196 | NTSTATUS status; |
| 197 | |
| 198 | RtlInitAnsiString( &str, name ); |
| 199 | pstrW = alloc ? &strW : &NtCurrentTeb()->StaticUnicodeString; |
| 200 | if (oem_file_apis) |
| 201 | status = RtlOemStringToUnicodeString( pstrW, &str, alloc ); |
| 202 | else |
| 203 | status = RtlAnsiStringToUnicodeString( pstrW, &str, alloc ); |
| 204 | if (status == STATUS_SUCCESS) return pstrW->Buffer; |
| 205 | |
| 206 | if (status == STATUS_BUFFER_OVERFLOW) |
| 207 | SetLastError( ERROR_FILENAME_EXCED_RANGE ); |
| 208 | else |
| 209 | SetLastError( RtlNtStatusToDosError(status) ); |
| 210 | return NULL; |
| 211 | } |
| 212 | |
| 213 | |
| 214 | /*********************************************************************** |
| 215 | * FILE_name_WtoA |
| 216 | * |
| 217 | * Convert a file name back to OEM/Ansi. Returns number of bytes copied. |
| 218 | */ |
| 219 | DWORD FILE_name_WtoA( LPCWSTR src, INT srclen, LPSTR dest, INT destlen ) |
| 220 | { |
| 221 | DWORD ret; |
| 222 | |
| 223 | if (srclen < 0) srclen = strlenW( src ) + 1; |
| 224 | if (oem_file_apis) |
| 225 | RtlUnicodeToOemN( dest, destlen, &ret, src, srclen * sizeof(WCHAR) ); |
| 226 | else |
| 227 | RtlUnicodeToMultiByteN( dest, destlen, &ret, src, srclen * sizeof(WCHAR) ); |
| 228 | return ret; |
| 229 | } |
| 230 | |
| 231 | |
| 232 | /************************************************************************** |
| 233 | * SetFileApisToOEM (KERNEL32.@) |
| 234 | */ |
| 235 | VOID WINAPI SetFileApisToOEM(void) |
| 236 | { |
| 237 | oem_file_apis = TRUE; |
| 238 | } |
| 239 | |
| 240 | |
| 241 | /************************************************************************** |
| 242 | * SetFileApisToANSI (KERNEL32.@) |
| 243 | */ |
| 244 | VOID WINAPI SetFileApisToANSI(void) |
| 245 | { |
| 246 | oem_file_apis = FALSE; |
| 247 | } |
| 248 | |
| 249 | |
| 250 | /****************************************************************************** |
| 251 | * AreFileApisANSI (KERNEL32.@) |
| 252 | * |
| 253 | * Determines if file functions are using ANSI |
| 254 | * |
| 255 | * RETURNS |
| 256 | * TRUE: Set of file functions is using ANSI code page |
| 257 | * FALSE: Set of file functions is using OEM code page |
| 258 | */ |
| 259 | BOOL WINAPI AreFileApisANSI(void) |
| 260 | { |
| 261 | return !oem_file_apis; |
| 262 | } |
| 263 | |
| 264 | |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 265 | /************************************************************************** |
| 266 | * Operations on file handles * |
| 267 | **************************************************************************/ |
| 268 | |
| 269 | /*********************************************************************** |
Eric Pouech | 768008f | 2004-01-14 04:34:20 +0000 | [diff] [blame] | 270 | * FILE_InitProcessDosHandles |
| 271 | * |
| 272 | * Allocates the default DOS handles for a process. Called either by |
| 273 | * Win32HandleToDosFileHandle below or by the DOSVM stuff. |
| 274 | */ |
| 275 | static void FILE_InitProcessDosHandles( void ) |
| 276 | { |
| 277 | static BOOL init_done /* = FALSE */; |
| 278 | HANDLE cp = GetCurrentProcess(); |
| 279 | |
| 280 | if (init_done) return; |
| 281 | init_done = TRUE; |
| 282 | DuplicateHandle(cp, GetStdHandle(STD_INPUT_HANDLE), cp, &dos_handles[0], |
| 283 | 0, TRUE, DUPLICATE_SAME_ACCESS); |
| 284 | DuplicateHandle(cp, GetStdHandle(STD_OUTPUT_HANDLE), cp, &dos_handles[1], |
| 285 | 0, TRUE, DUPLICATE_SAME_ACCESS); |
| 286 | DuplicateHandle(cp, GetStdHandle(STD_ERROR_HANDLE), cp, &dos_handles[2], |
| 287 | 0, TRUE, DUPLICATE_SAME_ACCESS); |
| 288 | DuplicateHandle(cp, GetStdHandle(STD_ERROR_HANDLE), cp, &dos_handles[3], |
| 289 | 0, TRUE, DUPLICATE_SAME_ACCESS); |
| 290 | DuplicateHandle(cp, GetStdHandle(STD_ERROR_HANDLE), cp, &dos_handles[4], |
| 291 | 0, TRUE, DUPLICATE_SAME_ACCESS); |
| 292 | } |
| 293 | |
| 294 | |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 295 | /****************************************************************** |
| 296 | * FILE_ReadWriteApc (internal) |
| 297 | */ |
| 298 | static void WINAPI FILE_ReadWriteApc(void* apc_user, PIO_STATUS_BLOCK io_status, ULONG len) |
| 299 | { |
| 300 | LPOVERLAPPED_COMPLETION_ROUTINE cr = (LPOVERLAPPED_COMPLETION_ROUTINE)apc_user; |
| 301 | |
| 302 | cr(RtlNtStatusToDosError(io_status->u.Status), len, (LPOVERLAPPED)io_status); |
| 303 | } |
| 304 | |
| 305 | |
| 306 | /*********************************************************************** |
| 307 | * ReadFileEx (KERNEL32.@) |
| 308 | */ |
| 309 | BOOL WINAPI ReadFileEx(HANDLE hFile, LPVOID buffer, DWORD bytesToRead, |
| 310 | LPOVERLAPPED overlapped, |
| 311 | LPOVERLAPPED_COMPLETION_ROUTINE lpCompletionRoutine) |
| 312 | { |
| 313 | LARGE_INTEGER offset; |
| 314 | NTSTATUS status; |
| 315 | PIO_STATUS_BLOCK io_status; |
| 316 | |
Rob Shearman | 95dc472 | 2004-07-09 22:24:59 +0000 | [diff] [blame] | 317 | TRACE("(hFile=%p, buffer=%p, bytes=%lu, ovl=%p, ovl_fn=%p)\n", hFile, buffer, bytesToRead, overlapped, lpCompletionRoutine); |
| 318 | |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 319 | if (!overlapped) |
| 320 | { |
| 321 | SetLastError(ERROR_INVALID_PARAMETER); |
| 322 | return FALSE; |
| 323 | } |
| 324 | |
| 325 | offset.u.LowPart = overlapped->Offset; |
| 326 | offset.u.HighPart = overlapped->OffsetHigh; |
| 327 | io_status = (PIO_STATUS_BLOCK)overlapped; |
| 328 | io_status->u.Status = STATUS_PENDING; |
| 329 | |
| 330 | status = NtReadFile(hFile, NULL, FILE_ReadWriteApc, lpCompletionRoutine, |
| 331 | io_status, buffer, bytesToRead, &offset, NULL); |
| 332 | |
| 333 | if (status) |
| 334 | { |
| 335 | SetLastError( RtlNtStatusToDosError(status) ); |
| 336 | return FALSE; |
| 337 | } |
| 338 | return TRUE; |
| 339 | } |
| 340 | |
| 341 | |
| 342 | /*********************************************************************** |
| 343 | * ReadFile (KERNEL32.@) |
| 344 | */ |
| 345 | BOOL WINAPI ReadFile( HANDLE hFile, LPVOID buffer, DWORD bytesToRead, |
| 346 | LPDWORD bytesRead, LPOVERLAPPED overlapped ) |
| 347 | { |
| 348 | LARGE_INTEGER offset; |
| 349 | PLARGE_INTEGER poffset = NULL; |
| 350 | IO_STATUS_BLOCK iosb; |
| 351 | PIO_STATUS_BLOCK io_status = &iosb; |
| 352 | HANDLE hEvent = 0; |
| 353 | NTSTATUS status; |
| 354 | |
| 355 | TRACE("%p %p %ld %p %p\n", hFile, buffer, bytesToRead, |
| 356 | bytesRead, overlapped ); |
| 357 | |
| 358 | if (bytesRead) *bytesRead = 0; /* Do this before anything else */ |
| 359 | if (!bytesToRead) return TRUE; |
| 360 | |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 361 | if (is_console_handle(hFile)) |
| 362 | return ReadConsoleA(hFile, buffer, bytesToRead, bytesRead, NULL); |
| 363 | |
| 364 | if (overlapped != NULL) |
| 365 | { |
| 366 | offset.u.LowPart = overlapped->Offset; |
| 367 | offset.u.HighPart = overlapped->OffsetHigh; |
| 368 | poffset = &offset; |
| 369 | hEvent = overlapped->hEvent; |
| 370 | io_status = (PIO_STATUS_BLOCK)overlapped; |
| 371 | } |
| 372 | io_status->u.Status = STATUS_PENDING; |
| 373 | io_status->Information = 0; |
| 374 | |
| 375 | status = NtReadFile(hFile, hEvent, NULL, NULL, io_status, buffer, bytesToRead, poffset, NULL); |
| 376 | |
| 377 | if (status != STATUS_PENDING && bytesRead) |
| 378 | *bytesRead = io_status->Information; |
| 379 | |
| 380 | if (status && status != STATUS_END_OF_FILE) |
| 381 | { |
| 382 | SetLastError( RtlNtStatusToDosError(status) ); |
| 383 | return FALSE; |
| 384 | } |
| 385 | return TRUE; |
| 386 | } |
| 387 | |
| 388 | |
| 389 | /*********************************************************************** |
| 390 | * WriteFileEx (KERNEL32.@) |
| 391 | */ |
| 392 | BOOL WINAPI WriteFileEx(HANDLE hFile, LPCVOID buffer, DWORD bytesToWrite, |
| 393 | LPOVERLAPPED overlapped, |
| 394 | LPOVERLAPPED_COMPLETION_ROUTINE lpCompletionRoutine) |
| 395 | { |
| 396 | LARGE_INTEGER offset; |
| 397 | NTSTATUS status; |
| 398 | PIO_STATUS_BLOCK io_status; |
| 399 | |
| 400 | TRACE("%p %p %ld %p %p\n", hFile, buffer, bytesToWrite, overlapped, lpCompletionRoutine); |
| 401 | |
| 402 | if (overlapped == NULL) |
| 403 | { |
| 404 | SetLastError(ERROR_INVALID_PARAMETER); |
| 405 | return FALSE; |
| 406 | } |
| 407 | offset.u.LowPart = overlapped->Offset; |
| 408 | offset.u.HighPart = overlapped->OffsetHigh; |
| 409 | |
| 410 | io_status = (PIO_STATUS_BLOCK)overlapped; |
| 411 | io_status->u.Status = STATUS_PENDING; |
| 412 | |
| 413 | status = NtWriteFile(hFile, NULL, FILE_ReadWriteApc, lpCompletionRoutine, |
| 414 | io_status, buffer, bytesToWrite, &offset, NULL); |
| 415 | |
| 416 | if (status) SetLastError( RtlNtStatusToDosError(status) ); |
| 417 | return !status; |
| 418 | } |
| 419 | |
| 420 | |
| 421 | /*********************************************************************** |
| 422 | * WriteFile (KERNEL32.@) |
| 423 | */ |
| 424 | BOOL WINAPI WriteFile( HANDLE hFile, LPCVOID buffer, DWORD bytesToWrite, |
| 425 | LPDWORD bytesWritten, LPOVERLAPPED overlapped ) |
| 426 | { |
| 427 | HANDLE hEvent = NULL; |
| 428 | LARGE_INTEGER offset; |
| 429 | PLARGE_INTEGER poffset = NULL; |
| 430 | NTSTATUS status; |
| 431 | IO_STATUS_BLOCK iosb; |
| 432 | PIO_STATUS_BLOCK piosb = &iosb; |
| 433 | |
| 434 | TRACE("%p %p %ld %p %p\n", hFile, buffer, bytesToWrite, bytesWritten, overlapped ); |
| 435 | |
| 436 | if (is_console_handle(hFile)) |
| 437 | return WriteConsoleA(hFile, buffer, bytesToWrite, bytesWritten, NULL); |
| 438 | |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 439 | if (overlapped) |
| 440 | { |
| 441 | offset.u.LowPart = overlapped->Offset; |
| 442 | offset.u.HighPart = overlapped->OffsetHigh; |
| 443 | poffset = &offset; |
| 444 | hEvent = overlapped->hEvent; |
| 445 | piosb = (PIO_STATUS_BLOCK)overlapped; |
| 446 | } |
| 447 | piosb->u.Status = STATUS_PENDING; |
| 448 | piosb->Information = 0; |
| 449 | |
| 450 | status = NtWriteFile(hFile, hEvent, NULL, NULL, piosb, |
| 451 | buffer, bytesToWrite, poffset, NULL); |
Dmitry Timoshkov | e04da6b | 2004-11-28 14:58:11 +0000 | [diff] [blame] | 452 | |
| 453 | if (status != STATUS_PENDING && bytesWritten) |
| 454 | *bytesWritten = piosb->Information; |
| 455 | |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 456 | if (status) |
| 457 | { |
| 458 | SetLastError( RtlNtStatusToDosError(status) ); |
| 459 | return FALSE; |
| 460 | } |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 461 | return TRUE; |
| 462 | } |
| 463 | |
| 464 | |
Eric Pouech | 768008f | 2004-01-14 04:34:20 +0000 | [diff] [blame] | 465 | /*********************************************************************** |
Eric Pouech | 9bd4f6b | 2003-06-26 02:08:17 +0000 | [diff] [blame] | 466 | * GetOverlappedResult (KERNEL32.@) |
| 467 | * |
| 468 | * Check the result of an Asynchronous data transfer from a file. |
| 469 | * |
| 470 | * Parameters |
| 471 | * HANDLE hFile [in] handle of file to check on |
| 472 | * LPOVERLAPPED lpOverlapped [in/out] pointer to overlapped |
| 473 | * LPDWORD lpTransferred [in/out] number of bytes transferred |
| 474 | * BOOL bWait [in] wait for the transfer to complete ? |
| 475 | * |
| 476 | * RETURNS |
| 477 | * TRUE on success |
| 478 | * FALSE on failure |
| 479 | * |
| 480 | * If successful (and relevant) lpTransferred will hold the number of |
| 481 | * bytes transferred during the async operation. |
| 482 | * |
| 483 | * BUGS |
| 484 | * |
| 485 | * Currently only works for WaitCommEvent, ReadFile, WriteFile |
| 486 | * with communications ports. |
| 487 | * |
| 488 | */ |
| 489 | BOOL WINAPI GetOverlappedResult(HANDLE hFile, LPOVERLAPPED lpOverlapped, |
| 490 | LPDWORD lpTransferred, BOOL bWait) |
| 491 | { |
Eric Pouech | 1ffddb4 | 2004-08-17 23:37:55 +0000 | [diff] [blame] | 492 | DWORD r = WAIT_OBJECT_0; |
Eric Pouech | 9bd4f6b | 2003-06-26 02:08:17 +0000 | [diff] [blame] | 493 | |
Eric Pouech | 1ffddb4 | 2004-08-17 23:37:55 +0000 | [diff] [blame] | 494 | TRACE( "(%p %p %p %x)\n", hFile, lpOverlapped, lpTransferred, bWait ); |
Eric Pouech | 9bd4f6b | 2003-06-26 02:08:17 +0000 | [diff] [blame] | 495 | |
Eric Pouech | 1ffddb4 | 2004-08-17 23:37:55 +0000 | [diff] [blame] | 496 | if ( lpOverlapped == NULL ) |
Eric Pouech | 9bd4f6b | 2003-06-26 02:08:17 +0000 | [diff] [blame] | 497 | { |
| 498 | ERR("lpOverlapped was null\n"); |
| 499 | return FALSE; |
| 500 | } |
Eric Pouech | 9bd4f6b | 2003-06-26 02:08:17 +0000 | [diff] [blame] | 501 | if ( bWait ) |
| 502 | { |
Eric Pouech | 1ffddb4 | 2004-08-17 23:37:55 +0000 | [diff] [blame] | 503 | if ( lpOverlapped->hEvent ) |
| 504 | { |
| 505 | do |
| 506 | { |
| 507 | TRACE( "waiting on %p\n", lpOverlapped ); |
| 508 | r = WaitForSingleObjectEx( lpOverlapped->hEvent, INFINITE, TRUE ); |
| 509 | TRACE( "wait on %p returned %ld\n", lpOverlapped, r ); |
| 510 | } while ( r == WAIT_IO_COMPLETION ); |
| 511 | } |
| 512 | else |
| 513 | { |
| 514 | /* busy loop */ |
| 515 | while ( (volatile DWORD)lpOverlapped->Internal == STATUS_PENDING ) |
| 516 | Sleep( 10 ); |
| 517 | } |
Eric Pouech | 9bd4f6b | 2003-06-26 02:08:17 +0000 | [diff] [blame] | 518 | } |
| 519 | else if ( lpOverlapped->Internal == STATUS_PENDING ) |
| 520 | { |
| 521 | /* Wait in order to give APCs a chance to run. */ |
| 522 | /* This is cheating, so we must set the event again in case of success - |
| 523 | it may be a non-manual reset event. */ |
Eric Pouech | 1ffddb4 | 2004-08-17 23:37:55 +0000 | [diff] [blame] | 524 | do |
| 525 | { |
| 526 | TRACE( "waiting on %p\n", lpOverlapped ); |
| 527 | r = WaitForSingleObjectEx( lpOverlapped->hEvent, 0, TRUE ); |
| 528 | TRACE( "wait on %p returned %ld\n", lpOverlapped, r ); |
| 529 | } while ( r == WAIT_IO_COMPLETION ); |
| 530 | if ( r == WAIT_OBJECT_0 && lpOverlapped->hEvent ) |
| 531 | NtSetEvent( lpOverlapped->hEvent, NULL ); |
Eric Pouech | 9bd4f6b | 2003-06-26 02:08:17 +0000 | [diff] [blame] | 532 | } |
Eric Pouech | 1ffddb4 | 2004-08-17 23:37:55 +0000 | [diff] [blame] | 533 | if ( r == WAIT_FAILED ) |
| 534 | { |
| 535 | ERR("wait operation failed\n"); |
| 536 | return FALSE; |
| 537 | } |
| 538 | if (lpTransferred) *lpTransferred = lpOverlapped->InternalHigh; |
Eric Pouech | 9bd4f6b | 2003-06-26 02:08:17 +0000 | [diff] [blame] | 539 | |
| 540 | switch ( lpOverlapped->Internal ) |
| 541 | { |
| 542 | case STATUS_SUCCESS: |
| 543 | return TRUE; |
| 544 | case STATUS_PENDING: |
Eric Pouech | 1ffddb4 | 2004-08-17 23:37:55 +0000 | [diff] [blame] | 545 | SetLastError( ERROR_IO_INCOMPLETE ); |
| 546 | if ( bWait ) ERR("PENDING status after waiting!\n"); |
Eric Pouech | 9bd4f6b | 2003-06-26 02:08:17 +0000 | [diff] [blame] | 547 | return FALSE; |
| 548 | default: |
Eric Pouech | 1ffddb4 | 2004-08-17 23:37:55 +0000 | [diff] [blame] | 549 | SetLastError( RtlNtStatusToDosError( lpOverlapped->Internal ) ); |
Eric Pouech | 9bd4f6b | 2003-06-26 02:08:17 +0000 | [diff] [blame] | 550 | return FALSE; |
| 551 | } |
| 552 | } |
| 553 | |
| 554 | /*********************************************************************** |
| 555 | * CancelIo (KERNEL32.@) |
| 556 | */ |
| 557 | BOOL WINAPI CancelIo(HANDLE handle) |
| 558 | { |
| 559 | async_private *ovp,*t; |
| 560 | |
| 561 | TRACE("handle = %p\n",handle); |
| 562 | |
| 563 | for (ovp = NtCurrentTeb()->pending_list; ovp; ovp = t) |
| 564 | { |
| 565 | t = ovp->next; |
| 566 | if ( ovp->handle == handle ) |
| 567 | cancel_async ( ovp ); |
| 568 | } |
Alexandre Julliard | 2b72be5 | 2003-06-30 02:10:19 +0000 | [diff] [blame] | 569 | SleepEx(1,TRUE); |
Eric Pouech | 9bd4f6b | 2003-06-26 02:08:17 +0000 | [diff] [blame] | 570 | return TRUE; |
| 571 | } |
| 572 | |
| 573 | /*********************************************************************** |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 574 | * _hread (KERNEL32.@) |
| 575 | */ |
| 576 | LONG WINAPI _hread( HFILE hFile, LPVOID buffer, LONG count) |
| 577 | { |
| 578 | return _lread( hFile, buffer, count ); |
| 579 | } |
| 580 | |
| 581 | |
| 582 | /*********************************************************************** |
| 583 | * _hwrite (KERNEL32.@) |
| 584 | * |
| 585 | * experimentation yields that _lwrite: |
| 586 | * o truncates the file at the current position with |
| 587 | * a 0 len write |
| 588 | * o returns 0 on a 0 length write |
| 589 | * o works with console handles |
| 590 | * |
| 591 | */ |
| 592 | LONG WINAPI _hwrite( HFILE handle, LPCSTR buffer, LONG count ) |
| 593 | { |
| 594 | DWORD result; |
| 595 | |
| 596 | TRACE("%d %p %ld\n", handle, buffer, count ); |
| 597 | |
| 598 | if (!count) |
| 599 | { |
| 600 | /* Expand or truncate at current position */ |
| 601 | if (!SetEndOfFile( (HANDLE)handle )) return HFILE_ERROR; |
| 602 | return 0; |
| 603 | } |
| 604 | if (!WriteFile( (HANDLE)handle, buffer, count, &result, NULL )) |
| 605 | return HFILE_ERROR; |
| 606 | return result; |
| 607 | } |
| 608 | |
| 609 | |
| 610 | /*********************************************************************** |
| 611 | * _lclose (KERNEL32.@) |
| 612 | */ |
| 613 | HFILE WINAPI _lclose( HFILE hFile ) |
| 614 | { |
| 615 | TRACE("handle %d\n", hFile ); |
| 616 | return CloseHandle( (HANDLE)hFile ) ? 0 : HFILE_ERROR; |
| 617 | } |
| 618 | |
| 619 | |
| 620 | /*********************************************************************** |
| 621 | * _lcreat (KERNEL32.@) |
| 622 | */ |
| 623 | HFILE WINAPI _lcreat( LPCSTR path, INT attr ) |
| 624 | { |
| 625 | /* Mask off all flags not explicitly allowed by the doc */ |
| 626 | attr &= FILE_ATTRIBUTE_READONLY | FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_SYSTEM; |
| 627 | TRACE("%s %02x\n", path, attr ); |
| 628 | return (HFILE)CreateFileA( path, GENERIC_READ | GENERIC_WRITE, |
| 629 | FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, |
| 630 | CREATE_ALWAYS, attr, 0 ); |
| 631 | } |
| 632 | |
| 633 | |
| 634 | /*********************************************************************** |
| 635 | * _lopen (KERNEL32.@) |
| 636 | */ |
| 637 | HFILE WINAPI _lopen( LPCSTR path, INT mode ) |
| 638 | { |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 639 | TRACE("(%s,%04x)\n", debugstr_a(path), mode ); |
Alexandre Julliard | 2f8fd10 | 2004-05-03 20:20:54 +0000 | [diff] [blame] | 640 | return (HFILE)create_file_OF( path, mode & ~OF_CREATE ); |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 641 | } |
| 642 | |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 643 | /*********************************************************************** |
| 644 | * _lread (KERNEL32.@) |
| 645 | */ |
| 646 | UINT WINAPI _lread( HFILE handle, LPVOID buffer, UINT count ) |
| 647 | { |
| 648 | DWORD result; |
Francois Gouget | 1614f91 | 2004-01-16 21:23:32 +0000 | [diff] [blame] | 649 | if (!ReadFile( (HANDLE)handle, buffer, count, &result, NULL )) |
| 650 | return HFILE_ERROR; |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 651 | return result; |
| 652 | } |
| 653 | |
| 654 | |
| 655 | /*********************************************************************** |
| 656 | * _llseek (KERNEL32.@) |
| 657 | */ |
| 658 | LONG WINAPI _llseek( HFILE hFile, LONG lOffset, INT nOrigin ) |
| 659 | { |
| 660 | return SetFilePointer( (HANDLE)hFile, lOffset, NULL, nOrigin ); |
| 661 | } |
| 662 | |
| 663 | |
| 664 | /*********************************************************************** |
| 665 | * _lwrite (KERNEL32.@) |
| 666 | */ |
| 667 | UINT WINAPI _lwrite( HFILE hFile, LPCSTR buffer, UINT count ) |
| 668 | { |
| 669 | return (UINT)_hwrite( hFile, buffer, (LONG)count ); |
| 670 | } |
| 671 | |
| 672 | |
| 673 | /*********************************************************************** |
| 674 | * FlushFileBuffers (KERNEL32.@) |
| 675 | */ |
| 676 | BOOL WINAPI FlushFileBuffers( HANDLE hFile ) |
| 677 | { |
| 678 | NTSTATUS nts; |
| 679 | IO_STATUS_BLOCK ioblk; |
| 680 | |
| 681 | if (is_console_handle( hFile )) |
| 682 | { |
| 683 | /* this will fail (as expected) for an output handle */ |
Dmitry Timoshkov | a8f025f | 2004-12-22 14:51:46 +0000 | [diff] [blame] | 684 | return FlushConsoleInputBuffer( hFile ); |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 685 | } |
| 686 | nts = NtFlushBuffersFile( hFile, &ioblk ); |
| 687 | if (nts != STATUS_SUCCESS) |
| 688 | { |
| 689 | SetLastError( RtlNtStatusToDosError( nts ) ); |
| 690 | return FALSE; |
| 691 | } |
| 692 | |
| 693 | return TRUE; |
| 694 | } |
| 695 | |
| 696 | |
| 697 | /*********************************************************************** |
Alexandre Julliard | b592cbb | 2004-05-01 02:44:00 +0000 | [diff] [blame] | 698 | * GetFileType (KERNEL32.@) |
| 699 | */ |
| 700 | DWORD WINAPI GetFileType( HANDLE hFile ) |
| 701 | { |
| 702 | FILE_FS_DEVICE_INFORMATION info; |
| 703 | IO_STATUS_BLOCK io; |
| 704 | NTSTATUS status; |
| 705 | |
| 706 | if (is_console_handle( hFile )) return FILE_TYPE_CHAR; |
| 707 | |
| 708 | status = NtQueryVolumeInformationFile( hFile, &io, &info, sizeof(info), FileFsDeviceInformation ); |
| 709 | if (status != STATUS_SUCCESS) |
| 710 | { |
| 711 | SetLastError( RtlNtStatusToDosError(status) ); |
| 712 | return FILE_TYPE_UNKNOWN; |
| 713 | } |
| 714 | |
| 715 | switch(info.DeviceType) |
| 716 | { |
| 717 | case FILE_DEVICE_NULL: |
| 718 | case FILE_DEVICE_SERIAL_PORT: |
| 719 | case FILE_DEVICE_PARALLEL_PORT: |
| 720 | case FILE_DEVICE_UNKNOWN: |
| 721 | return FILE_TYPE_CHAR; |
| 722 | case FILE_DEVICE_NAMED_PIPE: |
| 723 | return FILE_TYPE_PIPE; |
| 724 | default: |
| 725 | return FILE_TYPE_DISK; |
| 726 | } |
| 727 | } |
| 728 | |
| 729 | |
| 730 | /*********************************************************************** |
Alexandre Julliard | d4874d6 | 2004-04-30 21:03:09 +0000 | [diff] [blame] | 731 | * GetFileInformationByHandle (KERNEL32.@) |
| 732 | */ |
| 733 | BOOL WINAPI GetFileInformationByHandle( HANDLE hFile, BY_HANDLE_FILE_INFORMATION *info ) |
| 734 | { |
| 735 | FILE_ALL_INFORMATION all_info; |
| 736 | IO_STATUS_BLOCK io; |
| 737 | NTSTATUS status; |
| 738 | |
| 739 | status = NtQueryInformationFile( hFile, &io, &all_info, sizeof(all_info), FileAllInformation ); |
| 740 | if (status == STATUS_SUCCESS) |
| 741 | { |
| 742 | info->dwFileAttributes = all_info.BasicInformation.FileAttributes; |
| 743 | info->ftCreationTime.dwHighDateTime = all_info.BasicInformation.CreationTime.u.HighPart; |
| 744 | info->ftCreationTime.dwLowDateTime = all_info.BasicInformation.CreationTime.u.LowPart; |
| 745 | info->ftLastAccessTime.dwHighDateTime = all_info.BasicInformation.LastAccessTime.u.HighPart; |
| 746 | info->ftLastAccessTime.dwLowDateTime = all_info.BasicInformation.LastAccessTime.u.LowPart; |
| 747 | info->ftLastWriteTime.dwHighDateTime = all_info.BasicInformation.LastWriteTime.u.HighPart; |
| 748 | info->ftLastWriteTime.dwLowDateTime = all_info.BasicInformation.LastWriteTime.u.LowPart; |
| 749 | info->dwVolumeSerialNumber = 0; /* FIXME */ |
| 750 | info->nFileSizeHigh = all_info.StandardInformation.EndOfFile.u.HighPart; |
| 751 | info->nFileSizeLow = all_info.StandardInformation.EndOfFile.u.LowPart; |
| 752 | info->nNumberOfLinks = all_info.StandardInformation.NumberOfLinks; |
| 753 | info->nFileIndexHigh = all_info.InternalInformation.IndexNumber.u.HighPart; |
| 754 | info->nFileIndexLow = all_info.InternalInformation.IndexNumber.u.LowPart; |
| 755 | return TRUE; |
| 756 | } |
| 757 | SetLastError( RtlNtStatusToDosError(status) ); |
| 758 | return FALSE; |
| 759 | } |
| 760 | |
| 761 | |
| 762 | /*********************************************************************** |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 763 | * GetFileSize (KERNEL32.@) |
| 764 | */ |
| 765 | DWORD WINAPI GetFileSize( HANDLE hFile, LPDWORD filesizehigh ) |
| 766 | { |
Alexandre Julliard | d4874d6 | 2004-04-30 21:03:09 +0000 | [diff] [blame] | 767 | LARGE_INTEGER size; |
| 768 | if (!GetFileSizeEx( hFile, &size )) return INVALID_FILE_SIZE; |
| 769 | if (filesizehigh) *filesizehigh = size.u.HighPart; |
| 770 | if (size.u.LowPart == INVALID_FILE_SIZE) SetLastError(0); |
| 771 | return size.u.LowPart; |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 772 | } |
| 773 | |
| 774 | |
| 775 | /*********************************************************************** |
| 776 | * GetFileSizeEx (KERNEL32.@) |
| 777 | */ |
| 778 | BOOL WINAPI GetFileSizeEx( HANDLE hFile, PLARGE_INTEGER lpFileSize ) |
| 779 | { |
Alexandre Julliard | d4874d6 | 2004-04-30 21:03:09 +0000 | [diff] [blame] | 780 | FILE_END_OF_FILE_INFORMATION info; |
| 781 | IO_STATUS_BLOCK io; |
| 782 | NTSTATUS status; |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 783 | |
Alexandre Julliard | d4874d6 | 2004-04-30 21:03:09 +0000 | [diff] [blame] | 784 | status = NtQueryInformationFile( hFile, &io, &info, sizeof(info), FileEndOfFileInformation ); |
| 785 | if (status == STATUS_SUCCESS) |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 786 | { |
Alexandre Julliard | d4874d6 | 2004-04-30 21:03:09 +0000 | [diff] [blame] | 787 | *lpFileSize = info.EndOfFile; |
| 788 | return TRUE; |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 789 | } |
Alexandre Julliard | d4874d6 | 2004-04-30 21:03:09 +0000 | [diff] [blame] | 790 | SetLastError( RtlNtStatusToDosError(status) ); |
| 791 | return FALSE; |
| 792 | } |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 793 | |
Alexandre Julliard | d4874d6 | 2004-04-30 21:03:09 +0000 | [diff] [blame] | 794 | |
Alexandre Julliard | b592cbb | 2004-05-01 02:44:00 +0000 | [diff] [blame] | 795 | /************************************************************************** |
| 796 | * SetEndOfFile (KERNEL32.@) |
| 797 | */ |
| 798 | BOOL WINAPI SetEndOfFile( HANDLE hFile ) |
| 799 | { |
| 800 | FILE_POSITION_INFORMATION pos; |
| 801 | FILE_END_OF_FILE_INFORMATION eof; |
| 802 | IO_STATUS_BLOCK io; |
| 803 | NTSTATUS status; |
| 804 | |
| 805 | status = NtQueryInformationFile( hFile, &io, &pos, sizeof(pos), FilePositionInformation ); |
| 806 | if (status == STATUS_SUCCESS) |
| 807 | { |
| 808 | eof.EndOfFile = pos.CurrentByteOffset; |
| 809 | status = NtSetInformationFile( hFile, &io, &eof, sizeof(eof), FileEndOfFileInformation ); |
| 810 | } |
| 811 | if (status == STATUS_SUCCESS) return TRUE; |
| 812 | SetLastError( RtlNtStatusToDosError(status) ); |
| 813 | return FALSE; |
| 814 | } |
| 815 | |
| 816 | |
Alexandre Julliard | d4874d6 | 2004-04-30 21:03:09 +0000 | [diff] [blame] | 817 | /*********************************************************************** |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 818 | * SetFilePointer (KERNEL32.@) |
| 819 | */ |
Alexandre Julliard | 410db2a | 2004-06-14 19:40:44 +0000 | [diff] [blame] | 820 | DWORD WINAPI SetFilePointer( HANDLE hFile, LONG distance, LONG *highword, DWORD method ) |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 821 | { |
Alexandre Julliard | 410db2a | 2004-06-14 19:40:44 +0000 | [diff] [blame] | 822 | LARGE_INTEGER dist, newpos; |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 823 | |
Alexandre Julliard | 410db2a | 2004-06-14 19:40:44 +0000 | [diff] [blame] | 824 | if (highword) |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 825 | { |
Alexandre Julliard | 410db2a | 2004-06-14 19:40:44 +0000 | [diff] [blame] | 826 | dist.u.LowPart = distance; |
| 827 | dist.u.HighPart = *highword; |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 828 | } |
Alexandre Julliard | 410db2a | 2004-06-14 19:40:44 +0000 | [diff] [blame] | 829 | else dist.QuadPart = distance; |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 830 | |
Alexandre Julliard | 410db2a | 2004-06-14 19:40:44 +0000 | [diff] [blame] | 831 | if (!SetFilePointerEx( hFile, dist, &newpos, method )) return INVALID_SET_FILE_POINTER; |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 832 | |
Alexandre Julliard | 410db2a | 2004-06-14 19:40:44 +0000 | [diff] [blame] | 833 | if (highword) *highword = newpos.u.HighPart; |
| 834 | if (newpos.u.LowPart == INVALID_SET_FILE_POINTER) SetLastError( 0 ); |
| 835 | return newpos.u.LowPart; |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 836 | } |
| 837 | |
| 838 | |
| 839 | /*********************************************************************** |
Mike McCormack | 6122eeb | 2004-06-14 17:56:50 +0000 | [diff] [blame] | 840 | * SetFilePointerEx (KERNEL32.@) |
| 841 | */ |
| 842 | BOOL WINAPI SetFilePointerEx( HANDLE hFile, LARGE_INTEGER distance, |
| 843 | LARGE_INTEGER *newpos, DWORD method ) |
| 844 | { |
| 845 | static const int whence[3] = { SEEK_SET, SEEK_CUR, SEEK_END }; |
| 846 | BOOL ret = FALSE; |
| 847 | NTSTATUS status; |
| 848 | int fd; |
| 849 | |
| 850 | TRACE("handle %p offset %s newpos %p origin %ld\n", |
| 851 | hFile, wine_dbgstr_longlong(distance.QuadPart), newpos, method ); |
| 852 | |
| 853 | if (method > FILE_END) |
| 854 | { |
| 855 | SetLastError( ERROR_INVALID_PARAMETER ); |
| 856 | return ret; |
| 857 | } |
| 858 | |
Alexandre Julliard | 6a27b48 | 2004-08-18 00:04:58 +0000 | [diff] [blame] | 859 | if (!(status = wine_server_handle_to_fd( hFile, 0, &fd, NULL ))) |
Mike McCormack | 6122eeb | 2004-06-14 17:56:50 +0000 | [diff] [blame] | 860 | { |
| 861 | off_t pos, res; |
| 862 | |
| 863 | pos = distance.QuadPart; |
| 864 | if ((res = lseek( fd, pos, whence[method] )) == (off_t)-1) |
| 865 | { |
| 866 | /* also check EPERM due to SuSE7 2.2.16 lseek() EPERM kernel bug */ |
| 867 | if (((errno == EINVAL) || (errno == EPERM)) && (method != FILE_BEGIN) && (pos < 0)) |
| 868 | SetLastError( ERROR_NEGATIVE_SEEK ); |
| 869 | else |
| 870 | FILE_SetDosError(); |
| 871 | } |
| 872 | else |
| 873 | { |
| 874 | ret = TRUE; |
| 875 | if( newpos ) |
| 876 | newpos->QuadPart = res; |
| 877 | } |
| 878 | wine_server_release_fd( hFile, fd ); |
| 879 | } |
| 880 | else SetLastError( RtlNtStatusToDosError(status) ); |
| 881 | |
| 882 | return ret; |
| 883 | } |
| 884 | |
| 885 | /*********************************************************************** |
Alexandre Julliard | d4874d6 | 2004-04-30 21:03:09 +0000 | [diff] [blame] | 886 | * GetFileTime (KERNEL32.@) |
| 887 | */ |
| 888 | BOOL WINAPI GetFileTime( HANDLE hFile, FILETIME *lpCreationTime, |
| 889 | FILETIME *lpLastAccessTime, FILETIME *lpLastWriteTime ) |
| 890 | { |
| 891 | FILE_BASIC_INFORMATION info; |
| 892 | IO_STATUS_BLOCK io; |
| 893 | NTSTATUS status; |
| 894 | |
| 895 | status = NtQueryInformationFile( hFile, &io, &info, sizeof(info), FileBasicInformation ); |
| 896 | if (status == STATUS_SUCCESS) |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 897 | { |
Alexandre Julliard | d4874d6 | 2004-04-30 21:03:09 +0000 | [diff] [blame] | 898 | if (lpCreationTime) |
| 899 | { |
| 900 | lpCreationTime->dwHighDateTime = info.CreationTime.u.HighPart; |
| 901 | lpCreationTime->dwLowDateTime = info.CreationTime.u.LowPart; |
| 902 | } |
| 903 | if (lpLastAccessTime) |
| 904 | { |
| 905 | lpLastAccessTime->dwHighDateTime = info.LastAccessTime.u.HighPart; |
| 906 | lpLastAccessTime->dwLowDateTime = info.LastAccessTime.u.LowPart; |
| 907 | } |
| 908 | if (lpLastWriteTime) |
| 909 | { |
| 910 | lpLastWriteTime->dwHighDateTime = info.LastWriteTime.u.HighPart; |
| 911 | lpLastWriteTime->dwLowDateTime = info.LastWriteTime.u.LowPart; |
| 912 | } |
| 913 | return TRUE; |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 914 | } |
Alexandre Julliard | d4874d6 | 2004-04-30 21:03:09 +0000 | [diff] [blame] | 915 | SetLastError( RtlNtStatusToDosError(status) ); |
| 916 | return FALSE; |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 917 | } |
| 918 | |
| 919 | |
Alexandre Julliard | 174e2a6 | 2004-04-20 00:36:29 +0000 | [diff] [blame] | 920 | /*********************************************************************** |
| 921 | * SetFileTime (KERNEL32.@) |
| 922 | */ |
| 923 | BOOL WINAPI SetFileTime( HANDLE hFile, const FILETIME *ctime, |
| 924 | const FILETIME *atime, const FILETIME *mtime ) |
| 925 | { |
| 926 | FILE_BASIC_INFORMATION info; |
| 927 | IO_STATUS_BLOCK io; |
| 928 | NTSTATUS status; |
| 929 | |
| 930 | memset( &info, 0, sizeof(info) ); |
| 931 | if (ctime) |
| 932 | { |
| 933 | info.CreationTime.u.HighPart = ctime->dwHighDateTime; |
| 934 | info.CreationTime.u.LowPart = ctime->dwLowDateTime; |
| 935 | } |
| 936 | if (atime) |
| 937 | { |
| 938 | info.LastAccessTime.u.HighPart = atime->dwHighDateTime; |
| 939 | info.LastAccessTime.u.LowPart = atime->dwLowDateTime; |
| 940 | } |
| 941 | if (mtime) |
| 942 | { |
| 943 | info.LastWriteTime.u.HighPart = mtime->dwHighDateTime; |
| 944 | info.LastWriteTime.u.LowPart = mtime->dwLowDateTime; |
| 945 | } |
| 946 | |
| 947 | status = NtSetInformationFile( hFile, &io, &info, sizeof(info), FileBasicInformation ); |
| 948 | if (status == STATUS_SUCCESS) return TRUE; |
| 949 | SetLastError( RtlNtStatusToDosError(status) ); |
| 950 | return FALSE; |
| 951 | } |
| 952 | |
| 953 | |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 954 | /************************************************************************** |
| 955 | * LockFile (KERNEL32.@) |
| 956 | */ |
| 957 | BOOL WINAPI LockFile( HANDLE hFile, DWORD offset_low, DWORD offset_high, |
| 958 | DWORD count_low, DWORD count_high ) |
| 959 | { |
| 960 | NTSTATUS status; |
| 961 | LARGE_INTEGER count, offset; |
| 962 | |
| 963 | TRACE( "%p %lx%08lx %lx%08lx\n", |
| 964 | hFile, offset_high, offset_low, count_high, count_low ); |
| 965 | |
Ge van Geldorp | 399901e | 2004-01-23 01:51:33 +0000 | [diff] [blame] | 966 | count.u.LowPart = count_low; |
| 967 | count.u.HighPart = count_high; |
| 968 | offset.u.LowPart = offset_low; |
| 969 | offset.u.HighPart = offset_high; |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 970 | |
| 971 | status = NtLockFile( hFile, 0, NULL, NULL, |
| 972 | NULL, &offset, &count, NULL, TRUE, TRUE ); |
| 973 | |
| 974 | if (status != STATUS_SUCCESS) SetLastError( RtlNtStatusToDosError(status) ); |
| 975 | return !status; |
| 976 | } |
| 977 | |
| 978 | |
| 979 | /************************************************************************** |
| 980 | * LockFileEx [KERNEL32.@] |
| 981 | * |
| 982 | * Locks a byte range within an open file for shared or exclusive access. |
| 983 | * |
| 984 | * RETURNS |
| 985 | * success: TRUE |
| 986 | * failure: FALSE |
| 987 | * |
| 988 | * NOTES |
| 989 | * Per Microsoft docs, the third parameter (reserved) must be set to 0. |
| 990 | */ |
| 991 | BOOL WINAPI LockFileEx( HANDLE hFile, DWORD flags, DWORD reserved, |
| 992 | DWORD count_low, DWORD count_high, LPOVERLAPPED overlapped ) |
| 993 | { |
| 994 | NTSTATUS status; |
| 995 | LARGE_INTEGER count, offset; |
| 996 | |
| 997 | if (reserved) |
| 998 | { |
| 999 | SetLastError( ERROR_INVALID_PARAMETER ); |
| 1000 | return FALSE; |
| 1001 | } |
| 1002 | |
| 1003 | TRACE( "%p %lx%08lx %lx%08lx flags %lx\n", |
| 1004 | hFile, overlapped->OffsetHigh, overlapped->Offset, |
| 1005 | count_high, count_low, flags ); |
| 1006 | |
Ge van Geldorp | 399901e | 2004-01-23 01:51:33 +0000 | [diff] [blame] | 1007 | count.u.LowPart = count_low; |
| 1008 | count.u.HighPart = count_high; |
| 1009 | offset.u.LowPart = overlapped->Offset; |
| 1010 | offset.u.HighPart = overlapped->OffsetHigh; |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 1011 | |
| 1012 | status = NtLockFile( hFile, overlapped->hEvent, NULL, NULL, |
| 1013 | NULL, &offset, &count, NULL, |
| 1014 | flags & LOCKFILE_FAIL_IMMEDIATELY, |
| 1015 | flags & LOCKFILE_EXCLUSIVE_LOCK ); |
| 1016 | |
| 1017 | if (status) SetLastError( RtlNtStatusToDosError(status) ); |
| 1018 | return !status; |
| 1019 | } |
| 1020 | |
| 1021 | |
| 1022 | /************************************************************************** |
| 1023 | * UnlockFile (KERNEL32.@) |
| 1024 | */ |
| 1025 | BOOL WINAPI UnlockFile( HANDLE hFile, DWORD offset_low, DWORD offset_high, |
| 1026 | DWORD count_low, DWORD count_high ) |
| 1027 | { |
| 1028 | NTSTATUS status; |
| 1029 | LARGE_INTEGER count, offset; |
| 1030 | |
Ge van Geldorp | 399901e | 2004-01-23 01:51:33 +0000 | [diff] [blame] | 1031 | count.u.LowPart = count_low; |
| 1032 | count.u.HighPart = count_high; |
| 1033 | offset.u.LowPart = offset_low; |
| 1034 | offset.u.HighPart = offset_high; |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 1035 | |
| 1036 | status = NtUnlockFile( hFile, NULL, &offset, &count, NULL); |
| 1037 | if (status) SetLastError( RtlNtStatusToDosError(status) ); |
| 1038 | return !status; |
| 1039 | } |
| 1040 | |
| 1041 | |
| 1042 | /************************************************************************** |
| 1043 | * UnlockFileEx (KERNEL32.@) |
| 1044 | */ |
| 1045 | BOOL WINAPI UnlockFileEx( HANDLE hFile, DWORD reserved, DWORD count_low, DWORD count_high, |
| 1046 | LPOVERLAPPED overlapped ) |
| 1047 | { |
| 1048 | if (reserved) |
| 1049 | { |
| 1050 | SetLastError( ERROR_INVALID_PARAMETER ); |
| 1051 | return FALSE; |
| 1052 | } |
| 1053 | if (overlapped->hEvent) FIXME("Unimplemented overlapped operation\n"); |
| 1054 | |
| 1055 | return UnlockFile( hFile, overlapped->Offset, overlapped->OffsetHigh, count_low, count_high ); |
| 1056 | } |
| 1057 | |
Eric Pouech | 768008f | 2004-01-14 04:34:20 +0000 | [diff] [blame] | 1058 | |
| 1059 | /*********************************************************************** |
| 1060 | * Win32HandleToDosFileHandle (KERNEL32.21) |
| 1061 | * |
| 1062 | * Allocate a DOS handle for a Win32 handle. The Win32 handle is no |
| 1063 | * longer valid after this function (even on failure). |
| 1064 | * |
| 1065 | * Note: this is not exactly right, since on Win95 the Win32 handles |
| 1066 | * are on top of DOS handles and we do it the other way |
| 1067 | * around. Should be good enough though. |
| 1068 | */ |
| 1069 | HFILE WINAPI Win32HandleToDosFileHandle( HANDLE handle ) |
| 1070 | { |
| 1071 | int i; |
| 1072 | |
| 1073 | if (!handle || (handle == INVALID_HANDLE_VALUE)) |
| 1074 | return HFILE_ERROR; |
| 1075 | |
| 1076 | FILE_InitProcessDosHandles(); |
| 1077 | for (i = 0; i < DOS_TABLE_SIZE; i++) |
| 1078 | if (!dos_handles[i]) |
| 1079 | { |
| 1080 | dos_handles[i] = handle; |
| 1081 | TRACE("Got %d for h32 %p\n", i, handle ); |
| 1082 | return (HFILE)i; |
| 1083 | } |
| 1084 | CloseHandle( handle ); |
| 1085 | SetLastError( ERROR_TOO_MANY_OPEN_FILES ); |
| 1086 | return HFILE_ERROR; |
| 1087 | } |
| 1088 | |
| 1089 | |
| 1090 | /*********************************************************************** |
| 1091 | * DosFileHandleToWin32Handle (KERNEL32.20) |
| 1092 | * |
| 1093 | * Return the Win32 handle for a DOS handle. |
| 1094 | * |
| 1095 | * Note: this is not exactly right, since on Win95 the Win32 handles |
| 1096 | * are on top of DOS handles and we do it the other way |
| 1097 | * around. Should be good enough though. |
| 1098 | */ |
| 1099 | HANDLE WINAPI DosFileHandleToWin32Handle( HFILE handle ) |
| 1100 | { |
| 1101 | HFILE16 hfile = (HFILE16)handle; |
| 1102 | if (hfile < 5) FILE_InitProcessDosHandles(); |
| 1103 | if ((hfile >= DOS_TABLE_SIZE) || !dos_handles[hfile]) |
| 1104 | { |
| 1105 | SetLastError( ERROR_INVALID_HANDLE ); |
| 1106 | return INVALID_HANDLE_VALUE; |
| 1107 | } |
| 1108 | return dos_handles[hfile]; |
| 1109 | } |
| 1110 | |
| 1111 | |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 1112 | /************************************************************************* |
| 1113 | * SetHandleCount (KERNEL32.@) |
| 1114 | */ |
| 1115 | UINT WINAPI SetHandleCount( UINT count ) |
| 1116 | { |
| 1117 | return min( 256, count ); |
| 1118 | } |
| 1119 | |
| 1120 | |
Eric Pouech | 768008f | 2004-01-14 04:34:20 +0000 | [diff] [blame] | 1121 | /*********************************************************************** |
| 1122 | * DisposeLZ32Handle (KERNEL32.22) |
| 1123 | * |
| 1124 | * Note: this is not entirely correct, we should only close the |
| 1125 | * 32-bit handle and not the 16-bit one, but we cannot do |
| 1126 | * this because of the way our DOS handles are implemented. |
| 1127 | * It shouldn't break anything though. |
| 1128 | */ |
| 1129 | void WINAPI DisposeLZ32Handle( HANDLE handle ) |
| 1130 | { |
| 1131 | int i; |
| 1132 | |
| 1133 | if (!handle || (handle == INVALID_HANDLE_VALUE)) return; |
| 1134 | |
| 1135 | for (i = 5; i < DOS_TABLE_SIZE; i++) |
| 1136 | if (dos_handles[i] == handle) |
| 1137 | { |
| 1138 | dos_handles[i] = 0; |
| 1139 | CloseHandle( handle ); |
| 1140 | break; |
| 1141 | } |
| 1142 | } |
| 1143 | |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 1144 | /************************************************************************** |
| 1145 | * Operations on file names * |
| 1146 | **************************************************************************/ |
| 1147 | |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 1148 | |
| 1149 | /************************************************************************* |
| 1150 | * CreateFileW [KERNEL32.@] Creates or opens a file or other object |
| 1151 | * |
| 1152 | * Creates or opens an object, and returns a handle that can be used to |
| 1153 | * access that object. |
| 1154 | * |
| 1155 | * PARAMS |
| 1156 | * |
| 1157 | * filename [in] pointer to filename to be accessed |
| 1158 | * access [in] access mode requested |
| 1159 | * sharing [in] share mode |
| 1160 | * sa [in] pointer to security attributes |
| 1161 | * creation [in] how to create the file |
| 1162 | * attributes [in] attributes for newly created file |
| 1163 | * template [in] handle to file with extended attributes to copy |
| 1164 | * |
| 1165 | * RETURNS |
| 1166 | * Success: Open handle to specified file |
| 1167 | * Failure: INVALID_HANDLE_VALUE |
| 1168 | */ |
| 1169 | HANDLE WINAPI CreateFileW( LPCWSTR filename, DWORD access, DWORD sharing, |
| 1170 | LPSECURITY_ATTRIBUTES sa, DWORD creation, |
| 1171 | DWORD attributes, HANDLE template ) |
| 1172 | { |
| 1173 | NTSTATUS status; |
| 1174 | UINT options; |
| 1175 | OBJECT_ATTRIBUTES attr; |
| 1176 | UNICODE_STRING nameW; |
| 1177 | IO_STATUS_BLOCK io; |
| 1178 | HANDLE ret; |
| 1179 | DWORD dosdev; |
| 1180 | static const WCHAR bkslashes_with_dotW[] = {'\\','\\','.','\\',0}; |
| 1181 | static const WCHAR coninW[] = {'C','O','N','I','N','$',0}; |
| 1182 | static const WCHAR conoutW[] = {'C','O','N','O','U','T','$',0}; |
| 1183 | |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 1184 | static const UINT nt_disposition[5] = |
| 1185 | { |
| 1186 | FILE_CREATE, /* CREATE_NEW */ |
| 1187 | FILE_OVERWRITE_IF, /* CREATE_ALWAYS */ |
| 1188 | FILE_OPEN, /* OPEN_EXISTING */ |
| 1189 | FILE_OPEN_IF, /* OPEN_ALWAYS */ |
| 1190 | FILE_OVERWRITE /* TRUNCATE_EXISTING */ |
| 1191 | }; |
| 1192 | |
| 1193 | |
| 1194 | /* sanity checks */ |
| 1195 | |
| 1196 | if (!filename || !filename[0]) |
| 1197 | { |
| 1198 | SetLastError( ERROR_PATH_NOT_FOUND ); |
| 1199 | return INVALID_HANDLE_VALUE; |
| 1200 | } |
| 1201 | |
Dmitry Timoshkov | 567bed8 | 2004-12-21 16:12:49 +0000 | [diff] [blame] | 1202 | TRACE("%s %s%s%s%s%s%s creation %ld attributes 0x%lx\n", debugstr_w(filename), |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 1203 | (access & GENERIC_READ)?"GENERIC_READ ":"", |
| 1204 | (access & GENERIC_WRITE)?"GENERIC_WRITE ":"", |
| 1205 | (!access)?"QUERY_ACCESS ":"", |
| 1206 | (sharing & FILE_SHARE_READ)?"FILE_SHARE_READ ":"", |
| 1207 | (sharing & FILE_SHARE_WRITE)?"FILE_SHARE_WRITE ":"", |
| 1208 | (sharing & FILE_SHARE_DELETE)?"FILE_SHARE_DELETE ":"", |
Dmitry Timoshkov | 567bed8 | 2004-12-21 16:12:49 +0000 | [diff] [blame] | 1209 | creation, attributes); |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 1210 | |
| 1211 | /* Open a console for CONIN$ or CONOUT$ */ |
| 1212 | |
| 1213 | if (!strcmpiW(filename, coninW) || !strcmpiW(filename, conoutW)) |
| 1214 | { |
| 1215 | ret = OpenConsoleW(filename, access, (sa && sa->bInheritHandle), creation); |
| 1216 | goto done; |
| 1217 | } |
| 1218 | |
| 1219 | if (!strncmpW(filename, bkslashes_with_dotW, 4)) |
| 1220 | { |
| 1221 | static const WCHAR pipeW[] = {'P','I','P','E','\\',0}; |
| 1222 | |
| 1223 | if ((isalphaW(filename[4]) && filename[5] == ':' && filename[6] == '\0') || |
| 1224 | !strncmpiW( filename + 4, pipeW, 5 )) |
| 1225 | { |
| 1226 | dosdev = 0; |
| 1227 | } |
| 1228 | else if ((dosdev = RtlIsDosDeviceName_U( filename + 4 ))) |
| 1229 | { |
| 1230 | dosdev += MAKELONG( 0, 4*sizeof(WCHAR) ); /* adjust position to start of filename */ |
| 1231 | } |
| 1232 | else if (filename[4]) |
| 1233 | { |
| 1234 | ret = VXD_Open( filename+4, access, sa ); |
| 1235 | goto done; |
| 1236 | } |
| 1237 | else |
| 1238 | { |
| 1239 | SetLastError( ERROR_INVALID_NAME ); |
| 1240 | return INVALID_HANDLE_VALUE; |
| 1241 | } |
| 1242 | } |
| 1243 | else dosdev = RtlIsDosDeviceName_U( filename ); |
| 1244 | |
| 1245 | if (dosdev) |
| 1246 | { |
| 1247 | static const WCHAR conW[] = {'C','O','N'}; |
| 1248 | |
| 1249 | if (LOWORD(dosdev) == sizeof(conW) && |
| 1250 | !memicmpW( filename + HIWORD(dosdev)/sizeof(WCHAR), conW, sizeof(conW))) |
| 1251 | { |
| 1252 | switch (access & (GENERIC_READ|GENERIC_WRITE)) |
| 1253 | { |
| 1254 | case GENERIC_READ: |
| 1255 | ret = OpenConsoleW(coninW, access, (sa && sa->bInheritHandle), creation); |
| 1256 | goto done; |
| 1257 | case GENERIC_WRITE: |
| 1258 | ret = OpenConsoleW(conoutW, access, (sa && sa->bInheritHandle), creation); |
| 1259 | goto done; |
| 1260 | default: |
| 1261 | SetLastError( ERROR_FILE_NOT_FOUND ); |
| 1262 | return INVALID_HANDLE_VALUE; |
| 1263 | } |
| 1264 | } |
| 1265 | } |
| 1266 | |
Dmitry Timoshkov | 567bed8 | 2004-12-21 16:12:49 +0000 | [diff] [blame] | 1267 | if (creation < CREATE_NEW || creation > TRUNCATE_EXISTING) |
| 1268 | { |
| 1269 | SetLastError( ERROR_INVALID_PARAMETER ); |
| 1270 | return INVALID_HANDLE_VALUE; |
| 1271 | } |
| 1272 | |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 1273 | if (!RtlDosPathNameToNtPathName_U( filename, &nameW, NULL, NULL )) |
| 1274 | { |
| 1275 | SetLastError( ERROR_PATH_NOT_FOUND ); |
| 1276 | return INVALID_HANDLE_VALUE; |
| 1277 | } |
| 1278 | |
| 1279 | /* now call NtCreateFile */ |
| 1280 | |
| 1281 | options = 0; |
| 1282 | if (attributes & FILE_FLAG_BACKUP_SEMANTICS) |
| 1283 | options |= FILE_OPEN_FOR_BACKUP_INTENT; |
| 1284 | else |
| 1285 | options |= FILE_NON_DIRECTORY_FILE; |
| 1286 | if (attributes & FILE_FLAG_DELETE_ON_CLOSE) |
| 1287 | options |= FILE_DELETE_ON_CLOSE; |
| 1288 | if (!(attributes & FILE_FLAG_OVERLAPPED)) |
| 1289 | options |= FILE_SYNCHRONOUS_IO_ALERT; |
| 1290 | if (attributes & FILE_FLAG_RANDOM_ACCESS) |
| 1291 | options |= FILE_RANDOM_ACCESS; |
| 1292 | attributes &= FILE_ATTRIBUTE_VALID_FLAGS; |
| 1293 | |
| 1294 | attr.Length = sizeof(attr); |
| 1295 | attr.RootDirectory = 0; |
| 1296 | attr.Attributes = OBJ_CASE_INSENSITIVE; |
| 1297 | attr.ObjectName = &nameW; |
| 1298 | attr.SecurityDescriptor = sa ? sa->lpSecurityDescriptor : NULL; |
| 1299 | attr.SecurityQualityOfService = NULL; |
| 1300 | |
| 1301 | if (sa && sa->bInheritHandle) attr.Attributes |= OBJ_INHERIT; |
| 1302 | |
| 1303 | status = NtCreateFile( &ret, access, &attr, &io, NULL, attributes, |
| 1304 | sharing, nt_disposition[creation - CREATE_NEW], |
| 1305 | options, NULL, 0 ); |
| 1306 | if (status) |
| 1307 | { |
| 1308 | WARN("Unable to create file %s (status %lx)\n", debugstr_w(filename), status); |
| 1309 | ret = INVALID_HANDLE_VALUE; |
| 1310 | |
| 1311 | /* In the case file creation was rejected due to CREATE_NEW flag |
| 1312 | * was specified and file with that name already exists, correct |
| 1313 | * last error is ERROR_FILE_EXISTS and not ERROR_ALREADY_EXISTS. |
| 1314 | * Note: RtlNtStatusToDosError is not the subject to blame here. |
| 1315 | */ |
| 1316 | if (status == STATUS_OBJECT_NAME_COLLISION) |
| 1317 | SetLastError( ERROR_FILE_EXISTS ); |
| 1318 | else |
| 1319 | SetLastError( RtlNtStatusToDosError(status) ); |
| 1320 | } |
| 1321 | else SetLastError(0); |
| 1322 | RtlFreeUnicodeString( &nameW ); |
| 1323 | |
| 1324 | done: |
| 1325 | if (!ret) ret = INVALID_HANDLE_VALUE; |
| 1326 | TRACE("returning %p\n", ret); |
| 1327 | return ret; |
| 1328 | } |
| 1329 | |
| 1330 | |
| 1331 | |
| 1332 | /************************************************************************* |
| 1333 | * CreateFileA (KERNEL32.@) |
| 1334 | */ |
| 1335 | HANDLE WINAPI CreateFileA( LPCSTR filename, DWORD access, DWORD sharing, |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1336 | LPSECURITY_ATTRIBUTES sa, DWORD creation, |
| 1337 | DWORD attributes, HANDLE template) |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 1338 | { |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1339 | WCHAR *nameW; |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 1340 | |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1341 | if (!(nameW = FILE_name_AtoW( filename, FALSE ))) return INVALID_HANDLE_VALUE; |
| 1342 | return CreateFileW( nameW, access, sharing, sa, creation, attributes, template ); |
Alexandre Julliard | 643617f | 2004-05-01 05:25:07 +0000 | [diff] [blame] | 1343 | } |
| 1344 | |
| 1345 | |
Alexandre Julliard | c4c4da4 | 2004-04-03 00:05:24 +0000 | [diff] [blame] | 1346 | /*********************************************************************** |
| 1347 | * DeleteFileW (KERNEL32.@) |
| 1348 | */ |
| 1349 | BOOL WINAPI DeleteFileW( LPCWSTR path ) |
| 1350 | { |
Eric Pouech | 6d0712a | 2004-11-21 15:29:44 +0000 | [diff] [blame] | 1351 | UNICODE_STRING nameW; |
| 1352 | OBJECT_ATTRIBUTES attr; |
| 1353 | NTSTATUS status; |
Alexandre Julliard | c4c4da4 | 2004-04-03 00:05:24 +0000 | [diff] [blame] | 1354 | |
| 1355 | TRACE("%s\n", debugstr_w(path) ); |
| 1356 | |
Eric Pouech | 6d0712a | 2004-11-21 15:29:44 +0000 | [diff] [blame] | 1357 | if (!RtlDosPathNameToNtPathName_U( path, &nameW, NULL, NULL )) |
| 1358 | { |
| 1359 | SetLastError( ERROR_PATH_NOT_FOUND ); |
| 1360 | return FALSE; |
| 1361 | } |
Alexandre Julliard | c4c4da4 | 2004-04-03 00:05:24 +0000 | [diff] [blame] | 1362 | |
Eric Pouech | 6d0712a | 2004-11-21 15:29:44 +0000 | [diff] [blame] | 1363 | attr.Length = sizeof(attr); |
| 1364 | attr.RootDirectory = 0; |
| 1365 | attr.Attributes = OBJ_CASE_INSENSITIVE; |
| 1366 | attr.ObjectName = &nameW; |
| 1367 | attr.SecurityDescriptor = NULL; |
| 1368 | attr.SecurityQualityOfService = NULL; |
| 1369 | |
| 1370 | status = NtDeleteFile(&attr); |
| 1371 | RtlFreeUnicodeString( &nameW ); |
| 1372 | if (status) |
| 1373 | { |
| 1374 | SetLastError( RtlNtStatusToDosError(status) ); |
| 1375 | return FALSE; |
| 1376 | } |
Alexandre Julliard | c4c4da4 | 2004-04-03 00:05:24 +0000 | [diff] [blame] | 1377 | return TRUE; |
| 1378 | } |
| 1379 | |
| 1380 | |
| 1381 | /*********************************************************************** |
| 1382 | * DeleteFileA (KERNEL32.@) |
| 1383 | */ |
| 1384 | BOOL WINAPI DeleteFileA( LPCSTR path ) |
| 1385 | { |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1386 | WCHAR *pathW; |
Alexandre Julliard | c4c4da4 | 2004-04-03 00:05:24 +0000 | [diff] [blame] | 1387 | |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1388 | if (!(pathW = FILE_name_AtoW( path, FALSE ))) return FALSE; |
| 1389 | return DeleteFileW( pathW ); |
Alexandre Julliard | c4c4da4 | 2004-04-03 00:05:24 +0000 | [diff] [blame] | 1390 | } |
| 1391 | |
| 1392 | |
Eric Pouech | f6a7096 | 2003-06-24 02:32:01 +0000 | [diff] [blame] | 1393 | /************************************************************************** |
| 1394 | * ReplaceFileW (KERNEL32.@) |
| 1395 | * ReplaceFile (KERNEL32.@) |
| 1396 | */ |
| 1397 | BOOL WINAPI ReplaceFileW(LPCWSTR lpReplacedFileName,LPCWSTR lpReplacementFileName, |
| 1398 | LPCWSTR lpBackupFileName, DWORD dwReplaceFlags, |
| 1399 | LPVOID lpExclude, LPVOID lpReserved) |
| 1400 | { |
| 1401 | FIXME("(%s,%s,%s,%08lx,%p,%p) stub\n",debugstr_w(lpReplacedFileName),debugstr_w(lpReplacementFileName), |
| 1402 | debugstr_w(lpBackupFileName),dwReplaceFlags,lpExclude,lpReserved); |
| 1403 | SetLastError(ERROR_UNABLE_TO_MOVE_REPLACEMENT); |
| 1404 | return FALSE; |
| 1405 | } |
| 1406 | |
| 1407 | |
| 1408 | /************************************************************************** |
| 1409 | * ReplaceFileA (KERNEL32.@) |
| 1410 | */ |
| 1411 | BOOL WINAPI ReplaceFileA(LPCSTR lpReplacedFileName,LPCSTR lpReplacementFileName, |
| 1412 | LPCSTR lpBackupFileName, DWORD dwReplaceFlags, |
| 1413 | LPVOID lpExclude, LPVOID lpReserved) |
| 1414 | { |
| 1415 | FIXME("(%s,%s,%s,%08lx,%p,%p) stub\n",lpReplacedFileName,lpReplacementFileName, |
| 1416 | lpBackupFileName,dwReplaceFlags,lpExclude,lpReserved); |
| 1417 | SetLastError(ERROR_UNABLE_TO_MOVE_REPLACEMENT); |
| 1418 | return FALSE; |
| 1419 | } |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1420 | |
| 1421 | |
| 1422 | /************************************************************************* |
| 1423 | * FindFirstFileExW (KERNEL32.@) |
| 1424 | */ |
| 1425 | HANDLE WINAPI FindFirstFileExW( LPCWSTR filename, FINDEX_INFO_LEVELS level, |
| 1426 | LPVOID data, FINDEX_SEARCH_OPS search_op, |
| 1427 | LPVOID filter, DWORD flags) |
| 1428 | { |
Alexandre Julliard | a9832be | 2004-04-17 00:26:54 +0000 | [diff] [blame] | 1429 | WCHAR *mask, *p; |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1430 | FIND_FIRST_INFO *info = NULL; |
Alexandre Julliard | a9832be | 2004-04-17 00:26:54 +0000 | [diff] [blame] | 1431 | UNICODE_STRING nt_name; |
| 1432 | OBJECT_ATTRIBUTES attr; |
| 1433 | IO_STATUS_BLOCK io; |
| 1434 | NTSTATUS status; |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1435 | |
Christian Costa | fe0460f | 2004-05-22 03:13:56 +0000 | [diff] [blame] | 1436 | TRACE("%s %d %p %d %p %lx\n", debugstr_w(filename), level, data, search_op, filter, flags); |
| 1437 | |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1438 | if ((search_op != FindExSearchNameMatch) || (flags != 0)) |
| 1439 | { |
| 1440 | FIXME("options not implemented 0x%08x 0x%08lx\n", search_op, flags ); |
| 1441 | return INVALID_HANDLE_VALUE; |
| 1442 | } |
| 1443 | if (level != FindExInfoStandard) |
| 1444 | { |
| 1445 | FIXME("info level %d not implemented\n", level ); |
| 1446 | return INVALID_HANDLE_VALUE; |
| 1447 | } |
Alexandre Julliard | a9832be | 2004-04-17 00:26:54 +0000 | [diff] [blame] | 1448 | |
| 1449 | if (!RtlDosPathNameToNtPathName_U( filename, &nt_name, &mask, NULL )) |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1450 | { |
| 1451 | SetLastError( ERROR_PATH_NOT_FOUND ); |
| 1452 | return INVALID_HANDLE_VALUE; |
| 1453 | } |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1454 | |
| 1455 | if (!mask || !*mask) |
| 1456 | { |
| 1457 | SetLastError( ERROR_FILE_NOT_FOUND ); |
| 1458 | goto error; |
| 1459 | } |
| 1460 | |
| 1461 | if (!(info = HeapAlloc( GetProcessHeap(), 0, sizeof(*info)))) |
| 1462 | { |
| 1463 | SetLastError( ERROR_NOT_ENOUGH_MEMORY ); |
| 1464 | goto error; |
| 1465 | } |
| 1466 | |
| 1467 | if (!RtlCreateUnicodeString( &info->mask, mask )) |
| 1468 | { |
| 1469 | SetLastError( ERROR_NOT_ENOUGH_MEMORY ); |
| 1470 | goto error; |
| 1471 | } |
Alexandre Julliard | a9832be | 2004-04-17 00:26:54 +0000 | [diff] [blame] | 1472 | |
| 1473 | /* truncate dir name before mask */ |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1474 | *mask = 0; |
Alexandre Julliard | a9832be | 2004-04-17 00:26:54 +0000 | [diff] [blame] | 1475 | nt_name.Length = (mask - nt_name.Buffer) * sizeof(WCHAR); |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1476 | |
| 1477 | /* check if path is the root of the drive */ |
| 1478 | info->is_root = FALSE; |
Alexandre Julliard | a9832be | 2004-04-17 00:26:54 +0000 | [diff] [blame] | 1479 | p = nt_name.Buffer + 4; /* skip \??\ prefix */ |
| 1480 | if (p[0] && p[1] == ':') |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1481 | { |
Alexandre Julliard | a9832be | 2004-04-17 00:26:54 +0000 | [diff] [blame] | 1482 | p += 2; |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1483 | while (*p == '\\') p++; |
| 1484 | info->is_root = (*p == 0); |
| 1485 | } |
| 1486 | |
Alexandre Julliard | a9832be | 2004-04-17 00:26:54 +0000 | [diff] [blame] | 1487 | attr.Length = sizeof(attr); |
| 1488 | attr.RootDirectory = 0; |
| 1489 | attr.Attributes = OBJ_CASE_INSENSITIVE; |
| 1490 | attr.ObjectName = &nt_name; |
| 1491 | attr.SecurityDescriptor = NULL; |
| 1492 | attr.SecurityQualityOfService = NULL; |
| 1493 | |
| 1494 | status = NtOpenFile( &info->handle, GENERIC_READ, &attr, &io, |
| 1495 | FILE_SHARE_READ | FILE_SHARE_WRITE, |
| 1496 | FILE_DIRECTORY_FILE | FILE_SYNCHRONOUS_IO_NONALERT ); |
| 1497 | |
| 1498 | if (status != STATUS_SUCCESS) |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1499 | { |
| 1500 | RtlFreeUnicodeString( &info->mask ); |
Alexandre Julliard | a9832be | 2004-04-17 00:26:54 +0000 | [diff] [blame] | 1501 | SetLastError( RtlNtStatusToDosError(status) ); |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1502 | goto error; |
| 1503 | } |
Alexandre Julliard | a9832be | 2004-04-17 00:26:54 +0000 | [diff] [blame] | 1504 | RtlFreeUnicodeString( &nt_name ); |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1505 | |
| 1506 | RtlInitializeCriticalSection( &info->cs ); |
Alexandre Julliard | f435914 | 2004-06-15 00:55:04 +0000 | [diff] [blame] | 1507 | info->magic = FIND_FIRST_MAGIC; |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1508 | info->data_pos = 0; |
| 1509 | info->data_len = 0; |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1510 | |
| 1511 | if (!FindNextFileW( (HANDLE)info, data )) |
| 1512 | { |
| 1513 | TRACE( "%s not found\n", debugstr_w(filename) ); |
| 1514 | FindClose( (HANDLE)info ); |
| 1515 | SetLastError( ERROR_FILE_NOT_FOUND ); |
| 1516 | return INVALID_HANDLE_VALUE; |
| 1517 | } |
| 1518 | return (HANDLE)info; |
| 1519 | |
| 1520 | error: |
Michael Stefaniuc | 5ad7d85 | 2004-12-23 17:06:43 +0000 | [diff] [blame] | 1521 | HeapFree( GetProcessHeap(), 0, info ); |
Alexandre Julliard | a9832be | 2004-04-17 00:26:54 +0000 | [diff] [blame] | 1522 | RtlFreeUnicodeString( &nt_name ); |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1523 | return INVALID_HANDLE_VALUE; |
| 1524 | } |
| 1525 | |
| 1526 | |
| 1527 | /************************************************************************* |
| 1528 | * FindNextFileW (KERNEL32.@) |
| 1529 | */ |
| 1530 | BOOL WINAPI FindNextFileW( HANDLE handle, WIN32_FIND_DATAW *data ) |
| 1531 | { |
| 1532 | FIND_FIRST_INFO *info; |
| 1533 | FILE_BOTH_DIR_INFORMATION *dir_info; |
| 1534 | BOOL ret = FALSE; |
| 1535 | |
Christian Costa | fe0460f | 2004-05-22 03:13:56 +0000 | [diff] [blame] | 1536 | TRACE("%p %p\n", handle, data); |
Alexandre Julliard | f435914 | 2004-06-15 00:55:04 +0000 | [diff] [blame] | 1537 | |
| 1538 | if (!handle || handle == INVALID_HANDLE_VALUE) |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1539 | { |
| 1540 | SetLastError( ERROR_INVALID_HANDLE ); |
| 1541 | return ret; |
| 1542 | } |
| 1543 | info = (FIND_FIRST_INFO *)handle; |
Alexandre Julliard | f435914 | 2004-06-15 00:55:04 +0000 | [diff] [blame] | 1544 | if (info->magic != FIND_FIRST_MAGIC) |
| 1545 | { |
| 1546 | SetLastError( ERROR_INVALID_HANDLE ); |
| 1547 | return ret; |
| 1548 | } |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1549 | |
| 1550 | RtlEnterCriticalSection( &info->cs ); |
| 1551 | |
| 1552 | for (;;) |
| 1553 | { |
| 1554 | if (info->data_pos >= info->data_len) /* need to read some more data */ |
| 1555 | { |
| 1556 | IO_STATUS_BLOCK io; |
| 1557 | |
| 1558 | NtQueryDirectoryFile( info->handle, 0, NULL, NULL, &io, info->data, sizeof(info->data), |
| 1559 | FileBothDirectoryInformation, FALSE, &info->mask, FALSE ); |
| 1560 | if (io.u.Status) |
| 1561 | { |
| 1562 | SetLastError( RtlNtStatusToDosError( io.u.Status ) ); |
| 1563 | break; |
| 1564 | } |
| 1565 | info->data_len = io.Information; |
| 1566 | info->data_pos = 0; |
| 1567 | } |
| 1568 | |
| 1569 | dir_info = (FILE_BOTH_DIR_INFORMATION *)(info->data + info->data_pos); |
| 1570 | |
| 1571 | if (dir_info->NextEntryOffset) info->data_pos += dir_info->NextEntryOffset; |
| 1572 | else info->data_pos = info->data_len; |
| 1573 | |
| 1574 | /* don't return '.' and '..' in the root of the drive */ |
| 1575 | if (info->is_root) |
| 1576 | { |
| 1577 | if (dir_info->FileNameLength == sizeof(WCHAR) && dir_info->FileName[0] == '.') continue; |
| 1578 | if (dir_info->FileNameLength == 2 * sizeof(WCHAR) && |
| 1579 | dir_info->FileName[0] == '.' && dir_info->FileName[1] == '.') continue; |
| 1580 | } |
| 1581 | |
| 1582 | data->dwFileAttributes = dir_info->FileAttributes; |
| 1583 | data->ftCreationTime = *(FILETIME *)&dir_info->CreationTime; |
| 1584 | data->ftLastAccessTime = *(FILETIME *)&dir_info->LastAccessTime; |
| 1585 | data->ftLastWriteTime = *(FILETIME *)&dir_info->LastWriteTime; |
| 1586 | data->nFileSizeHigh = dir_info->EndOfFile.QuadPart >> 32; |
| 1587 | data->nFileSizeLow = (DWORD)dir_info->EndOfFile.QuadPart; |
| 1588 | data->dwReserved0 = 0; |
| 1589 | data->dwReserved1 = 0; |
| 1590 | |
| 1591 | memcpy( data->cFileName, dir_info->FileName, dir_info->FileNameLength ); |
| 1592 | data->cFileName[dir_info->FileNameLength/sizeof(WCHAR)] = 0; |
| 1593 | memcpy( data->cAlternateFileName, dir_info->ShortName, dir_info->ShortNameLength ); |
| 1594 | data->cAlternateFileName[dir_info->ShortNameLength/sizeof(WCHAR)] = 0; |
| 1595 | |
| 1596 | TRACE("returning %s (%s)\n", |
| 1597 | debugstr_w(data->cFileName), debugstr_w(data->cAlternateFileName) ); |
| 1598 | |
| 1599 | ret = TRUE; |
| 1600 | break; |
| 1601 | } |
| 1602 | |
| 1603 | RtlLeaveCriticalSection( &info->cs ); |
| 1604 | return ret; |
| 1605 | } |
| 1606 | |
| 1607 | |
| 1608 | /************************************************************************* |
| 1609 | * FindClose (KERNEL32.@) |
| 1610 | */ |
| 1611 | BOOL WINAPI FindClose( HANDLE handle ) |
| 1612 | { |
| 1613 | FIND_FIRST_INFO *info = (FIND_FIRST_INFO *)handle; |
| 1614 | |
Alexandre Julliard | f435914 | 2004-06-15 00:55:04 +0000 | [diff] [blame] | 1615 | if (!handle || handle == INVALID_HANDLE_VALUE) |
| 1616 | { |
| 1617 | SetLastError( ERROR_INVALID_HANDLE ); |
| 1618 | return FALSE; |
| 1619 | } |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1620 | |
| 1621 | __TRY |
| 1622 | { |
Alexandre Julliard | f435914 | 2004-06-15 00:55:04 +0000 | [diff] [blame] | 1623 | if (info->magic == FIND_FIRST_MAGIC) |
| 1624 | { |
| 1625 | RtlEnterCriticalSection( &info->cs ); |
| 1626 | if (info->magic == FIND_FIRST_MAGIC) /* in case someone else freed it in the meantime */ |
| 1627 | { |
| 1628 | info->magic = 0; |
| 1629 | if (info->handle) CloseHandle( info->handle ); |
| 1630 | info->handle = 0; |
| 1631 | RtlFreeUnicodeString( &info->mask ); |
| 1632 | info->mask.Buffer = NULL; |
| 1633 | info->data_pos = 0; |
| 1634 | info->data_len = 0; |
| 1635 | RtlLeaveCriticalSection( &info->cs ); |
| 1636 | RtlDeleteCriticalSection( &info->cs ); |
| 1637 | HeapFree( GetProcessHeap(), 0, info ); |
| 1638 | } |
| 1639 | } |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1640 | } |
| 1641 | __EXCEPT(page_fault) |
| 1642 | { |
| 1643 | WARN("Illegal handle %p\n", handle); |
| 1644 | SetLastError( ERROR_INVALID_HANDLE ); |
| 1645 | return FALSE; |
| 1646 | } |
| 1647 | __ENDTRY |
| 1648 | |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1649 | return TRUE; |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1650 | } |
| 1651 | |
| 1652 | |
| 1653 | /************************************************************************* |
| 1654 | * FindFirstFileA (KERNEL32.@) |
| 1655 | */ |
| 1656 | HANDLE WINAPI FindFirstFileA( LPCSTR lpFileName, WIN32_FIND_DATAA *lpFindData ) |
| 1657 | { |
| 1658 | return FindFirstFileExA(lpFileName, FindExInfoStandard, lpFindData, |
| 1659 | FindExSearchNameMatch, NULL, 0); |
| 1660 | } |
| 1661 | |
| 1662 | /************************************************************************* |
| 1663 | * FindFirstFileExA (KERNEL32.@) |
| 1664 | */ |
| 1665 | HANDLE WINAPI FindFirstFileExA( LPCSTR lpFileName, FINDEX_INFO_LEVELS fInfoLevelId, |
| 1666 | LPVOID lpFindFileData, FINDEX_SEARCH_OPS fSearchOp, |
| 1667 | LPVOID lpSearchFilter, DWORD dwAdditionalFlags) |
| 1668 | { |
| 1669 | HANDLE handle; |
| 1670 | WIN32_FIND_DATAA *dataA; |
| 1671 | WIN32_FIND_DATAW dataW; |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1672 | WCHAR *nameW; |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1673 | |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1674 | if (!(nameW = FILE_name_AtoW( lpFileName, FALSE ))) return INVALID_HANDLE_VALUE; |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1675 | |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1676 | handle = FindFirstFileExW(nameW, fInfoLevelId, &dataW, fSearchOp, lpSearchFilter, dwAdditionalFlags); |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1677 | if (handle == INVALID_HANDLE_VALUE) return handle; |
| 1678 | |
| 1679 | dataA = (WIN32_FIND_DATAA *) lpFindFileData; |
| 1680 | dataA->dwFileAttributes = dataW.dwFileAttributes; |
| 1681 | dataA->ftCreationTime = dataW.ftCreationTime; |
| 1682 | dataA->ftLastAccessTime = dataW.ftLastAccessTime; |
| 1683 | dataA->ftLastWriteTime = dataW.ftLastWriteTime; |
| 1684 | dataA->nFileSizeHigh = dataW.nFileSizeHigh; |
| 1685 | dataA->nFileSizeLow = dataW.nFileSizeLow; |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1686 | FILE_name_WtoA( dataW.cFileName, -1, dataA->cFileName, sizeof(dataA->cFileName) ); |
| 1687 | FILE_name_WtoA( dataW.cAlternateFileName, -1, dataA->cAlternateFileName, |
| 1688 | sizeof(dataA->cAlternateFileName) ); |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1689 | return handle; |
| 1690 | } |
| 1691 | |
| 1692 | |
| 1693 | /************************************************************************* |
| 1694 | * FindFirstFileW (KERNEL32.@) |
| 1695 | */ |
| 1696 | HANDLE WINAPI FindFirstFileW( LPCWSTR lpFileName, WIN32_FIND_DATAW *lpFindData ) |
| 1697 | { |
| 1698 | return FindFirstFileExW(lpFileName, FindExInfoStandard, lpFindData, |
| 1699 | FindExSearchNameMatch, NULL, 0); |
| 1700 | } |
| 1701 | |
| 1702 | |
| 1703 | /************************************************************************* |
| 1704 | * FindNextFileA (KERNEL32.@) |
| 1705 | */ |
| 1706 | BOOL WINAPI FindNextFileA( HANDLE handle, WIN32_FIND_DATAA *data ) |
| 1707 | { |
| 1708 | WIN32_FIND_DATAW dataW; |
| 1709 | |
| 1710 | if (!FindNextFileW( handle, &dataW )) return FALSE; |
| 1711 | data->dwFileAttributes = dataW.dwFileAttributes; |
| 1712 | data->ftCreationTime = dataW.ftCreationTime; |
| 1713 | data->ftLastAccessTime = dataW.ftLastAccessTime; |
| 1714 | data->ftLastWriteTime = dataW.ftLastWriteTime; |
| 1715 | data->nFileSizeHigh = dataW.nFileSizeHigh; |
| 1716 | data->nFileSizeLow = dataW.nFileSizeLow; |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1717 | FILE_name_WtoA( dataW.cFileName, -1, data->cFileName, sizeof(data->cFileName) ); |
| 1718 | FILE_name_WtoA( dataW.cAlternateFileName, -1, data->cAlternateFileName, |
| 1719 | sizeof(data->cAlternateFileName) ); |
Alexandre Julliard | 8f03216 | 2004-03-17 20:57:09 +0000 | [diff] [blame] | 1720 | return TRUE; |
| 1721 | } |
Alexandre Julliard | 5bfafc9 | 2004-03-25 05:36:08 +0000 | [diff] [blame] | 1722 | |
| 1723 | |
Alexandre Julliard | 174e2a6 | 2004-04-20 00:36:29 +0000 | [diff] [blame] | 1724 | /************************************************************************** |
| 1725 | * GetFileAttributesW (KERNEL32.@) |
| 1726 | */ |
| 1727 | DWORD WINAPI GetFileAttributesW( LPCWSTR name ) |
| 1728 | { |
| 1729 | FILE_BASIC_INFORMATION info; |
| 1730 | UNICODE_STRING nt_name; |
| 1731 | OBJECT_ATTRIBUTES attr; |
| 1732 | NTSTATUS status; |
| 1733 | |
Christian Costa | fe0460f | 2004-05-22 03:13:56 +0000 | [diff] [blame] | 1734 | TRACE("%s\n", debugstr_w(name)); |
| 1735 | |
Alexandre Julliard | 174e2a6 | 2004-04-20 00:36:29 +0000 | [diff] [blame] | 1736 | if (!RtlDosPathNameToNtPathName_U( name, &nt_name, NULL, NULL )) |
| 1737 | { |
| 1738 | SetLastError( ERROR_PATH_NOT_FOUND ); |
| 1739 | return INVALID_FILE_ATTRIBUTES; |
| 1740 | } |
| 1741 | |
| 1742 | attr.Length = sizeof(attr); |
| 1743 | attr.RootDirectory = 0; |
| 1744 | attr.Attributes = OBJ_CASE_INSENSITIVE; |
| 1745 | attr.ObjectName = &nt_name; |
| 1746 | attr.SecurityDescriptor = NULL; |
| 1747 | attr.SecurityQualityOfService = NULL; |
| 1748 | |
| 1749 | status = NtQueryAttributesFile( &attr, &info ); |
| 1750 | RtlFreeUnicodeString( &nt_name ); |
| 1751 | |
| 1752 | if (status == STATUS_SUCCESS) return info.FileAttributes; |
| 1753 | |
| 1754 | /* NtQueryAttributesFile fails on devices, but GetFileAttributesW succeeds */ |
| 1755 | if (RtlIsDosDeviceName_U( name )) return FILE_ATTRIBUTE_ARCHIVE; |
| 1756 | |
| 1757 | SetLastError( RtlNtStatusToDosError(status) ); |
| 1758 | return INVALID_FILE_ATTRIBUTES; |
| 1759 | } |
| 1760 | |
| 1761 | |
| 1762 | /************************************************************************** |
| 1763 | * GetFileAttributesA (KERNEL32.@) |
| 1764 | */ |
| 1765 | DWORD WINAPI GetFileAttributesA( LPCSTR name ) |
| 1766 | { |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1767 | WCHAR *nameW; |
Alexandre Julliard | 174e2a6 | 2004-04-20 00:36:29 +0000 | [diff] [blame] | 1768 | |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1769 | if (!(nameW = FILE_name_AtoW( name, FALSE ))) return INVALID_FILE_ATTRIBUTES; |
| 1770 | return GetFileAttributesW( nameW ); |
Alexandre Julliard | 174e2a6 | 2004-04-20 00:36:29 +0000 | [diff] [blame] | 1771 | } |
| 1772 | |
| 1773 | |
| 1774 | /************************************************************************** |
| 1775 | * SetFileAttributesW (KERNEL32.@) |
| 1776 | */ |
| 1777 | BOOL WINAPI SetFileAttributesW( LPCWSTR name, DWORD attributes ) |
| 1778 | { |
| 1779 | UNICODE_STRING nt_name; |
| 1780 | OBJECT_ATTRIBUTES attr; |
| 1781 | IO_STATUS_BLOCK io; |
| 1782 | NTSTATUS status; |
| 1783 | HANDLE handle; |
| 1784 | |
Christian Costa | fe0460f | 2004-05-22 03:13:56 +0000 | [diff] [blame] | 1785 | TRACE("%s %lx\n", debugstr_w(name), attributes); |
| 1786 | |
Alexandre Julliard | 174e2a6 | 2004-04-20 00:36:29 +0000 | [diff] [blame] | 1787 | if (!RtlDosPathNameToNtPathName_U( name, &nt_name, NULL, NULL )) |
| 1788 | { |
| 1789 | SetLastError( ERROR_PATH_NOT_FOUND ); |
| 1790 | return FALSE; |
| 1791 | } |
| 1792 | |
| 1793 | attr.Length = sizeof(attr); |
| 1794 | attr.RootDirectory = 0; |
| 1795 | attr.Attributes = OBJ_CASE_INSENSITIVE; |
| 1796 | attr.ObjectName = &nt_name; |
| 1797 | attr.SecurityDescriptor = NULL; |
| 1798 | attr.SecurityQualityOfService = NULL; |
| 1799 | |
| 1800 | status = NtOpenFile( &handle, 0, &attr, &io, 0, FILE_SYNCHRONOUS_IO_NONALERT ); |
| 1801 | RtlFreeUnicodeString( &nt_name ); |
| 1802 | |
| 1803 | if (status == STATUS_SUCCESS) |
| 1804 | { |
| 1805 | FILE_BASIC_INFORMATION info; |
| 1806 | |
| 1807 | memset( &info, 0, sizeof(info) ); |
| 1808 | info.FileAttributes = attributes | FILE_ATTRIBUTE_NORMAL; /* make sure it's not zero */ |
| 1809 | status = NtSetInformationFile( handle, &io, &info, sizeof(info), FileBasicInformation ); |
| 1810 | NtClose( handle ); |
| 1811 | } |
| 1812 | |
| 1813 | if (status == STATUS_SUCCESS) return TRUE; |
| 1814 | SetLastError( RtlNtStatusToDosError(status) ); |
| 1815 | return FALSE; |
| 1816 | } |
| 1817 | |
| 1818 | |
| 1819 | /************************************************************************** |
| 1820 | * SetFileAttributesA (KERNEL32.@) |
| 1821 | */ |
| 1822 | BOOL WINAPI SetFileAttributesA( LPCSTR name, DWORD attributes ) |
| 1823 | { |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1824 | WCHAR *nameW; |
Alexandre Julliard | 174e2a6 | 2004-04-20 00:36:29 +0000 | [diff] [blame] | 1825 | |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1826 | if (!(nameW = FILE_name_AtoW( name, FALSE ))) return FALSE; |
| 1827 | return SetFileAttributesW( nameW, attributes ); |
Alexandre Julliard | 174e2a6 | 2004-04-20 00:36:29 +0000 | [diff] [blame] | 1828 | } |
| 1829 | |
| 1830 | |
Alexandre Julliard | e440932 | 2004-04-20 04:06:49 +0000 | [diff] [blame] | 1831 | /************************************************************************** |
| 1832 | * GetFileAttributesExW (KERNEL32.@) |
| 1833 | */ |
| 1834 | BOOL WINAPI GetFileAttributesExW( LPCWSTR name, GET_FILEEX_INFO_LEVELS level, LPVOID ptr ) |
| 1835 | { |
| 1836 | FILE_NETWORK_OPEN_INFORMATION info; |
| 1837 | WIN32_FILE_ATTRIBUTE_DATA *data = ptr; |
| 1838 | UNICODE_STRING nt_name; |
| 1839 | OBJECT_ATTRIBUTES attr; |
| 1840 | NTSTATUS status; |
Christian Costa | fe0460f | 2004-05-22 03:13:56 +0000 | [diff] [blame] | 1841 | |
| 1842 | TRACE("%s %d %p\n", debugstr_w(name), level, ptr); |
Alexandre Julliard | e440932 | 2004-04-20 04:06:49 +0000 | [diff] [blame] | 1843 | |
| 1844 | if (level != GetFileExInfoStandard) |
| 1845 | { |
| 1846 | SetLastError( ERROR_INVALID_PARAMETER ); |
| 1847 | return FALSE; |
| 1848 | } |
| 1849 | |
| 1850 | if (!RtlDosPathNameToNtPathName_U( name, &nt_name, NULL, NULL )) |
| 1851 | { |
| 1852 | SetLastError( ERROR_PATH_NOT_FOUND ); |
| 1853 | return FALSE; |
| 1854 | } |
| 1855 | |
| 1856 | attr.Length = sizeof(attr); |
| 1857 | attr.RootDirectory = 0; |
| 1858 | attr.Attributes = OBJ_CASE_INSENSITIVE; |
| 1859 | attr.ObjectName = &nt_name; |
| 1860 | attr.SecurityDescriptor = NULL; |
| 1861 | attr.SecurityQualityOfService = NULL; |
| 1862 | |
| 1863 | status = NtQueryFullAttributesFile( &attr, &info ); |
| 1864 | RtlFreeUnicodeString( &nt_name ); |
| 1865 | |
| 1866 | if (status != STATUS_SUCCESS) |
| 1867 | { |
| 1868 | SetLastError( RtlNtStatusToDosError(status) ); |
| 1869 | return FALSE; |
| 1870 | } |
| 1871 | |
| 1872 | data->dwFileAttributes = info.FileAttributes; |
| 1873 | data->ftCreationTime.dwLowDateTime = info.CreationTime.u.LowPart; |
| 1874 | data->ftCreationTime.dwHighDateTime = info.CreationTime.u.HighPart; |
| 1875 | data->ftLastAccessTime.dwLowDateTime = info.LastAccessTime.u.LowPart; |
| 1876 | data->ftLastAccessTime.dwHighDateTime = info.LastAccessTime.u.HighPart; |
| 1877 | data->ftLastWriteTime.dwLowDateTime = info.LastWriteTime.u.LowPart; |
| 1878 | data->ftLastWriteTime.dwHighDateTime = info.LastWriteTime.u.HighPart; |
| 1879 | data->nFileSizeLow = info.EndOfFile.u.LowPart; |
| 1880 | data->nFileSizeHigh = info.EndOfFile.u.HighPart; |
| 1881 | return TRUE; |
| 1882 | } |
| 1883 | |
| 1884 | |
| 1885 | /************************************************************************** |
| 1886 | * GetFileAttributesExA (KERNEL32.@) |
| 1887 | */ |
| 1888 | BOOL WINAPI GetFileAttributesExA( LPCSTR name, GET_FILEEX_INFO_LEVELS level, LPVOID ptr ) |
| 1889 | { |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1890 | WCHAR *nameW; |
Alexandre Julliard | e440932 | 2004-04-20 04:06:49 +0000 | [diff] [blame] | 1891 | |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1892 | if (!(nameW = FILE_name_AtoW( name, FALSE ))) return FALSE; |
| 1893 | return GetFileAttributesExW( nameW, level, ptr ); |
Alexandre Julliard | e440932 | 2004-04-20 04:06:49 +0000 | [diff] [blame] | 1894 | } |
| 1895 | |
| 1896 | |
| 1897 | /****************************************************************************** |
| 1898 | * GetCompressedFileSizeW (KERNEL32.@) |
| 1899 | * |
| 1900 | * RETURNS |
| 1901 | * Success: Low-order doubleword of number of bytes |
| 1902 | * Failure: INVALID_FILE_SIZE |
| 1903 | */ |
| 1904 | DWORD WINAPI GetCompressedFileSizeW( |
| 1905 | LPCWSTR name, /* [in] Pointer to name of file */ |
| 1906 | LPDWORD size_high ) /* [out] Receives high-order doubleword of size */ |
| 1907 | { |
| 1908 | UNICODE_STRING nt_name; |
| 1909 | OBJECT_ATTRIBUTES attr; |
| 1910 | IO_STATUS_BLOCK io; |
| 1911 | NTSTATUS status; |
| 1912 | HANDLE handle; |
| 1913 | DWORD ret = INVALID_FILE_SIZE; |
| 1914 | |
Christian Costa | fe0460f | 2004-05-22 03:13:56 +0000 | [diff] [blame] | 1915 | TRACE("%s %p\n", debugstr_w(name), size_high); |
| 1916 | |
Alexandre Julliard | e440932 | 2004-04-20 04:06:49 +0000 | [diff] [blame] | 1917 | if (!RtlDosPathNameToNtPathName_U( name, &nt_name, NULL, NULL )) |
| 1918 | { |
| 1919 | SetLastError( ERROR_PATH_NOT_FOUND ); |
| 1920 | return INVALID_FILE_SIZE; |
| 1921 | } |
| 1922 | |
| 1923 | attr.Length = sizeof(attr); |
| 1924 | attr.RootDirectory = 0; |
| 1925 | attr.Attributes = OBJ_CASE_INSENSITIVE; |
| 1926 | attr.ObjectName = &nt_name; |
| 1927 | attr.SecurityDescriptor = NULL; |
| 1928 | attr.SecurityQualityOfService = NULL; |
| 1929 | |
| 1930 | status = NtOpenFile( &handle, 0, &attr, &io, 0, FILE_SYNCHRONOUS_IO_NONALERT ); |
| 1931 | RtlFreeUnicodeString( &nt_name ); |
| 1932 | |
| 1933 | if (status == STATUS_SUCCESS) |
| 1934 | { |
| 1935 | /* we don't support compressed files, simply return the file size */ |
| 1936 | ret = GetFileSize( handle, size_high ); |
| 1937 | NtClose( handle ); |
| 1938 | } |
| 1939 | else SetLastError( RtlNtStatusToDosError(status) ); |
| 1940 | |
| 1941 | return ret; |
| 1942 | } |
| 1943 | |
| 1944 | |
| 1945 | /****************************************************************************** |
| 1946 | * GetCompressedFileSizeA (KERNEL32.@) |
| 1947 | */ |
| 1948 | DWORD WINAPI GetCompressedFileSizeA( LPCSTR name, LPDWORD size_high ) |
| 1949 | { |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1950 | WCHAR *nameW; |
Alexandre Julliard | e440932 | 2004-04-20 04:06:49 +0000 | [diff] [blame] | 1951 | |
Alexandre Julliard | 0d33e5e | 2004-05-13 20:21:25 +0000 | [diff] [blame] | 1952 | if (!(nameW = FILE_name_AtoW( name, FALSE ))) return INVALID_FILE_SIZE; |
| 1953 | return GetCompressedFileSizeW( nameW, size_high ); |
Alexandre Julliard | e440932 | 2004-04-20 04:06:49 +0000 | [diff] [blame] | 1954 | } |
| 1955 | |
| 1956 | |
Alexandre Julliard | 5bfafc9 | 2004-03-25 05:36:08 +0000 | [diff] [blame] | 1957 | /*********************************************************************** |
| 1958 | * OpenFile (KERNEL32.@) |
| 1959 | */ |
| 1960 | HFILE WINAPI OpenFile( LPCSTR name, OFSTRUCT *ofs, UINT mode ) |
| 1961 | { |
| 1962 | HANDLE handle; |
| 1963 | FILETIME filetime; |
| 1964 | WORD filedatetime[2]; |
| 1965 | |
| 1966 | if (!ofs) return HFILE_ERROR; |
| 1967 | |
| 1968 | TRACE("%s %s %s %s%s%s%s%s%s%s%s%s\n",name, |
| 1969 | ((mode & 0x3 )==OF_READ)?"OF_READ": |
| 1970 | ((mode & 0x3 )==OF_WRITE)?"OF_WRITE": |
| 1971 | ((mode & 0x3 )==OF_READWRITE)?"OF_READWRITE":"unknown", |
| 1972 | ((mode & 0x70 )==OF_SHARE_COMPAT)?"OF_SHARE_COMPAT": |
| 1973 | ((mode & 0x70 )==OF_SHARE_DENY_NONE)?"OF_SHARE_DENY_NONE": |
| 1974 | ((mode & 0x70 )==OF_SHARE_DENY_READ)?"OF_SHARE_DENY_READ": |
| 1975 | ((mode & 0x70 )==OF_SHARE_DENY_WRITE)?"OF_SHARE_DENY_WRITE": |
| 1976 | ((mode & 0x70 )==OF_SHARE_EXCLUSIVE)?"OF_SHARE_EXCLUSIVE":"unknown", |
| 1977 | ((mode & OF_PARSE )==OF_PARSE)?"OF_PARSE ":"", |
| 1978 | ((mode & OF_DELETE )==OF_DELETE)?"OF_DELETE ":"", |
| 1979 | ((mode & OF_VERIFY )==OF_VERIFY)?"OF_VERIFY ":"", |
| 1980 | ((mode & OF_SEARCH )==OF_SEARCH)?"OF_SEARCH ":"", |
| 1981 | ((mode & OF_CANCEL )==OF_CANCEL)?"OF_CANCEL ":"", |
| 1982 | ((mode & OF_CREATE )==OF_CREATE)?"OF_CREATE ":"", |
| 1983 | ((mode & OF_PROMPT )==OF_PROMPT)?"OF_PROMPT ":"", |
| 1984 | ((mode & OF_EXIST )==OF_EXIST)?"OF_EXIST ":"", |
| 1985 | ((mode & OF_REOPEN )==OF_REOPEN)?"OF_REOPEN ":"" |
| 1986 | ); |
| 1987 | |
| 1988 | |
| 1989 | ofs->cBytes = sizeof(OFSTRUCT); |
| 1990 | ofs->nErrCode = 0; |
| 1991 | if (mode & OF_REOPEN) name = ofs->szPathName; |
| 1992 | |
| 1993 | if (!name) return HFILE_ERROR; |
| 1994 | |
| 1995 | TRACE("%s %04x\n", name, mode ); |
| 1996 | |
| 1997 | /* the watcom 10.6 IDE relies on a valid path returned in ofs->szPathName |
| 1998 | Are there any cases where getting the path here is wrong? |
| 1999 | Uwe Bonnes 1997 Apr 2 */ |
| 2000 | if (!GetFullPathNameA( name, sizeof(ofs->szPathName), ofs->szPathName, NULL )) goto error; |
| 2001 | |
| 2002 | /* OF_PARSE simply fills the structure */ |
| 2003 | |
| 2004 | if (mode & OF_PARSE) |
| 2005 | { |
| 2006 | ofs->fFixedDisk = (GetDriveTypeA( ofs->szPathName ) != DRIVE_REMOVABLE); |
| 2007 | TRACE("(%s): OF_PARSE, res = '%s'\n", name, ofs->szPathName ); |
| 2008 | return 0; |
| 2009 | } |
| 2010 | |
| 2011 | /* OF_CREATE is completely different from all other options, so |
| 2012 | handle it first */ |
| 2013 | |
| 2014 | if (mode & OF_CREATE) |
| 2015 | { |
Alexandre Julliard | 2f8fd10 | 2004-05-03 20:20:54 +0000 | [diff] [blame] | 2016 | if ((handle = create_file_OF( name, mode )) == INVALID_HANDLE_VALUE) |
Alexandre Julliard | 5bfafc9 | 2004-03-25 05:36:08 +0000 | [diff] [blame] | 2017 | goto error; |
| 2018 | } |
| 2019 | else |
| 2020 | { |
| 2021 | /* Now look for the file */ |
| 2022 | |
| 2023 | if (!SearchPathA( NULL, name, NULL, sizeof(ofs->szPathName), ofs->szPathName, NULL )) |
| 2024 | goto error; |
| 2025 | |
| 2026 | TRACE("found %s\n", debugstr_a(ofs->szPathName) ); |
| 2027 | |
| 2028 | if (mode & OF_DELETE) |
| 2029 | { |
| 2030 | if (!DeleteFileA( ofs->szPathName )) goto error; |
| 2031 | TRACE("(%s): OF_DELETE return = OK\n", name); |
| 2032 | return TRUE; |
| 2033 | } |
| 2034 | |
| 2035 | handle = (HANDLE)_lopen( ofs->szPathName, mode ); |
| 2036 | if (handle == INVALID_HANDLE_VALUE) goto error; |
| 2037 | |
| 2038 | GetFileTime( handle, NULL, NULL, &filetime ); |
| 2039 | FileTimeToDosDateTime( &filetime, &filedatetime[0], &filedatetime[1] ); |
| 2040 | if ((mode & OF_VERIFY) && (mode & OF_REOPEN)) |
| 2041 | { |
| 2042 | if (ofs->Reserved1 != filedatetime[0] || ofs->Reserved2 != filedatetime[1] ) |
| 2043 | { |
| 2044 | CloseHandle( handle ); |
| 2045 | WARN("(%s): OF_VERIFY failed\n", name ); |
| 2046 | /* FIXME: what error here? */ |
| 2047 | SetLastError( ERROR_FILE_NOT_FOUND ); |
| 2048 | goto error; |
| 2049 | } |
| 2050 | } |
| 2051 | ofs->Reserved1 = filedatetime[0]; |
| 2052 | ofs->Reserved2 = filedatetime[1]; |
| 2053 | } |
| 2054 | TRACE("(%s): OK, return = %p\n", name, handle ); |
| 2055 | if (mode & OF_EXIST) /* Return TRUE instead of a handle */ |
| 2056 | { |
| 2057 | CloseHandle( handle ); |
| 2058 | return TRUE; |
| 2059 | } |
| 2060 | else return (HFILE)handle; |
| 2061 | |
| 2062 | error: /* We get here if there was an error opening the file */ |
| 2063 | ofs->nErrCode = GetLastError(); |
| 2064 | WARN("(%s): return = HFILE_ERROR error= %d\n", name,ofs->nErrCode ); |
| 2065 | return HFILE_ERROR; |
| 2066 | } |