VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/win/path-win.cpp@ 36879

Last change on this file since 36879 was 36879, checked in by vboxsync, 14 years ago

IPRT/path-win: Fixed NT4 breakage when using SHGetFolderPathW.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 23.8 KB
Line 
1/* $Id: path-win.cpp 36879 2011-04-29 09:15:52Z vboxsync $ */
2/** @file
3 * IPRT - Path manipulation.
4 */
5
6/*
7 * Copyright (C) 2006-2010 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*******************************************************************************
29* Header Files *
30*******************************************************************************/
31#define LOG_GROUP RTLOGGROUP_PATH
32#include <Windows.h>
33#include <Shlobj.h>
34
35#include <iprt/path.h>
36#include <iprt/assert.h>
37#include <iprt/string.h>
38#include <iprt/time.h>
39#include <iprt/ldr.h>
40#include <iprt/mem.h>
41#include <iprt/param.h>
42#include <iprt/log.h>
43#include <iprt/err.h>
44#include "internal/path.h"
45#include "internal/fs.h"
46
47/* Needed for lazy loading SHGetFolderPathW in RTPathUserDocuments(). */
48typedef HRESULT FNSHGETFOLDERPATHW(HWND, int, HANDLE, DWORD, LPWSTR);
49typedef FNSHGETFOLDERPATHW *PFNSHGETFOLDERPATHW;
50
51/**
52 * Get the real (no symlinks, no . or .. components) path, must exist.
53 *
54 * @returns iprt status code.
55 * @param pszPath The path to resolve.
56 * @param pszRealPath Where to store the real path.
57 * @param cchRealPath Size of the buffer.
58 */
59RTDECL(int) RTPathReal(const char *pszPath, char *pszRealPath, size_t cchRealPath)
60{
61 /*
62 * Convert to UTF-16, call Win32 APIs, convert back.
63 */
64 PRTUTF16 pwszPath;
65 int rc = RTStrToUtf16(pszPath, &pwszPath);
66 if (!RT_SUCCESS(rc))
67 return (rc);
68
69 LPWSTR lpFile;
70 WCHAR wsz[RTPATH_MAX];
71 rc = GetFullPathNameW((LPCWSTR)pwszPath, RT_ELEMENTS(wsz), &wsz[0], &lpFile);
72 if (rc > 0 && rc < RT_ELEMENTS(wsz))
73 {
74 /* Check that it exists. (Use RTPathAbs() to just resolve the name.) */
75 DWORD dwAttr = GetFileAttributesW(wsz);
76 if (dwAttr != INVALID_FILE_ATTRIBUTES)
77 rc = RTUtf16ToUtf8Ex((PRTUTF16)&wsz[0], RTSTR_MAX, &pszRealPath, cchRealPath, NULL);
78 else
79 rc = RTErrConvertFromWin32(GetLastError());
80 }
81 else if (rc <= 0)
82 rc = RTErrConvertFromWin32(GetLastError());
83 else
84 rc = VERR_FILENAME_TOO_LONG;
85
86 RTUtf16Free(pwszPath);
87
88 return rc;
89}
90
91
92/**
93 * Get the absolute path (no symlinks, no . or .. components), doesn't have to exit.
94 *
95 * @returns iprt status code.
96 * @param pszPath The path to resolve.
97 * @param pszAbsPath Where to store the absolute path.
98 * @param cchAbsPath Size of the buffer.
99 */
100RTDECL(int) RTPathAbs(const char *pszPath, char *pszAbsPath, size_t cchAbsPath)
101{
102 /*
103 * Validation.
104 */
105 AssertPtr(pszAbsPath);
106 AssertPtr(pszPath);
107 if (RT_UNLIKELY(!*pszPath))
108 return VERR_INVALID_PARAMETER;
109
110 /*
111 * Convert to UTF-16, call Win32 API, convert back.
112 */
113 LPWSTR pwszPath;
114 int rc = RTStrToUtf16(pszPath, &pwszPath);
115 if (!RT_SUCCESS(rc))
116 return (rc);
117
118 LPWSTR pwszFile; /* Ignored */
119 RTUTF16 wsz[RTPATH_MAX];
120 rc = GetFullPathNameW(pwszPath, RT_ELEMENTS(wsz), &wsz[0], &pwszFile);
121 if (rc > 0 && rc < RT_ELEMENTS(wsz))
122 {
123 size_t cch;
124 rc = RTUtf16ToUtf8Ex(&wsz[0], RTSTR_MAX, &pszAbsPath, cchAbsPath, &cch);
125 if (RT_SUCCESS(rc))
126 {
127 /*
128 * Remove trailing slash if the path may be pointing to a directory.
129 * (See posix variant.)
130 */
131 if ( cch > 1
132 && RTPATH_IS_SLASH(pszAbsPath[cch - 1])
133 && !RTPATH_IS_VOLSEP(pszAbsPath[cch - 2])
134 && !RTPATH_IS_SLASH(pszAbsPath[cch - 2]))
135 pszAbsPath[cch - 1] = '\0';
136 }
137 }
138 else if (rc <= 0)
139 rc = RTErrConvertFromWin32(GetLastError());
140 else
141 rc = VERR_FILENAME_TOO_LONG;
142
143 RTUtf16Free(pwszPath);
144 return rc;
145}
146
147
148/**
149 * Gets the user home directory.
150 *
151 * @returns iprt status code.
152 * @param pszPath Buffer where to store the path.
153 * @param cchPath Buffer size in bytes.
154 */
155RTDECL(int) RTPathUserHome(char *pszPath, size_t cchPath)
156{
157 RTUTF16 wszPath[RTPATH_MAX];
158 DWORD dwAttr;
159
160 /*
161 * There are multiple definitions for what WE think of as user home...
162 */
163 if ( !GetEnvironmentVariableW(L"HOME", &wszPath[0], RTPATH_MAX)
164 || (dwAttr = GetFileAttributesW(&wszPath[0])) == INVALID_FILE_ATTRIBUTES
165 || !(dwAttr & FILE_ATTRIBUTE_DIRECTORY))
166 {
167 if ( !GetEnvironmentVariableW(L"USERPROFILE", &wszPath[0], RTPATH_MAX)
168 || (dwAttr = GetFileAttributesW(&wszPath[0])) == INVALID_FILE_ATTRIBUTES
169 || !(dwAttr & FILE_ATTRIBUTE_DIRECTORY))
170 {
171 /* %HOMEDRIVE%%HOMEPATH% */
172 if (!GetEnvironmentVariableW(L"HOMEDRIVE", &wszPath[0], RTPATH_MAX))
173 return VERR_PATH_NOT_FOUND;
174 size_t const cwc = RTUtf16Len(&wszPath[0]);
175 if ( !GetEnvironmentVariableW(L"HOMEPATH", &wszPath[cwc], RTPATH_MAX - (DWORD)cwc)
176 || (dwAttr = GetFileAttributesW(&wszPath[0])) == INVALID_FILE_ATTRIBUTES
177 || !(dwAttr & FILE_ATTRIBUTE_DIRECTORY))
178 return VERR_PATH_NOT_FOUND;
179 }
180 }
181
182 /*
183 * Convert and return.
184 */
185 return RTUtf16ToUtf8Ex(&wszPath[0], RTSTR_MAX, &pszPath, cchPath, NULL);
186}
187
188
189RTDECL(int) RTPathUserDocuments(char *pszPath, size_t cchPath)
190{
191 /*
192 * Validate input
193 */
194 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
195 AssertReturn(cchPath, VERR_INVALID_PARAMETER);
196
197 RTLDRMOD hShell32;
198 int rc = RTLdrLoad("Shell32.dll", &hShell32);
199 if (RT_SUCCESS(rc))
200 {
201 PFNSHGETFOLDERPATHW pfnSHGetFolderPathW;
202 rc = RTLdrGetSymbol(hShell32, "SHGetFolderPathW", (void**)&pfnSHGetFolderPathW);
203 if (RT_SUCCESS(rc))
204 {
205 RTUTF16 wszPath[RTPATH_MAX];
206 HRESULT hrc = pfnSHGetFolderPathW(0, CSIDL_PERSONAL, NULL, SHGFP_TYPE_CURRENT, wszPath);
207 if ( hrc == S_OK /* Found */
208 || hrc == S_FALSE) /* Found, but doesn't exist */
209 {
210 /*
211 * Convert and return.
212 */
213 RTLdrClose(hShell32);
214 return RTUtf16ToUtf8Ex(&wszPath[0], RTSTR_MAX, &pszPath, cchPath, NULL);
215 }
216 }
217 RTLdrClose(hShell32);
218 }
219 return VERR_PATH_NOT_FOUND;
220}
221
222
223RTR3DECL(int) RTPathQueryInfo(const char *pszPath, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAdditionalAttribs)
224{
225 return RTPathQueryInfoEx(pszPath, pObjInfo, enmAdditionalAttribs, RTPATH_F_ON_LINK);
226}
227
228
229RTR3DECL(int) RTPathQueryInfoEx(const char *pszPath, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAdditionalAttribs, uint32_t fFlags)
230{
231 /*
232 * Validate input.
233 */
234 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
235 AssertReturn(*pszPath, VERR_INVALID_PARAMETER);
236 AssertPtrReturn(pObjInfo, VERR_INVALID_POINTER);
237 AssertMsgReturn( enmAdditionalAttribs >= RTFSOBJATTRADD_NOTHING
238 && enmAdditionalAttribs <= RTFSOBJATTRADD_LAST,
239 ("Invalid enmAdditionalAttribs=%p\n", enmAdditionalAttribs),
240 VERR_INVALID_PARAMETER);
241 AssertMsgReturn(RTPATH_F_IS_VALID(fFlags, 0), ("%#x\n", fFlags), VERR_INVALID_PARAMETER);
242
243 /*
244 * Query file info.
245 */
246 WIN32_FILE_ATTRIBUTE_DATA Data;
247 PRTUTF16 pwszPath;
248 int rc = RTStrToUtf16(pszPath, &pwszPath);
249 if (RT_FAILURE(rc))
250 return rc;
251 if (!GetFileAttributesExW(pwszPath, GetFileExInfoStandard, &Data))
252 {
253 /* Fallback to FindFileFirst in case of sharing violation. */
254 if (GetLastError() == ERROR_SHARING_VIOLATION)
255 {
256 WIN32_FIND_DATAW FindData;
257 HANDLE hDir = FindFirstFileW(pwszPath, &FindData);
258 if (hDir == INVALID_HANDLE_VALUE)
259 {
260 rc = RTErrConvertFromWin32(GetLastError());
261 RTUtf16Free(pwszPath);
262 return rc;
263 }
264 FindClose(hDir);
265
266 Data.dwFileAttributes = FindData.dwFileAttributes;
267 Data.ftCreationTime = FindData.ftCreationTime;
268 Data.ftLastAccessTime = FindData.ftLastAccessTime;
269 Data.ftLastWriteTime = FindData.ftLastWriteTime;
270 Data.nFileSizeHigh = FindData.nFileSizeHigh;
271 Data.nFileSizeLow = FindData.nFileSizeLow;
272 }
273 else
274 {
275 rc = RTErrConvertFromWin32(GetLastError());
276 RTUtf16Free(pwszPath);
277 return rc;
278 }
279 }
280
281 /*
282 * Getting the information for the link target is a bit annoying and
283 * subject to the same access violation mess as above.. :/
284 */
285 /** @todo we're too lazy wrt to error paths here... */
286 if ( (fFlags & RTPATH_F_FOLLOW_LINK)
287 && (Data.dwFileAttributes & FILE_ATTRIBUTE_REPARSE_POINT))
288 {
289 HANDLE hFinal = CreateFileW(pwszPath,
290 GENERIC_READ,
291 FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
292 NULL,
293 OPEN_EXISTING,
294 FILE_FLAG_BACKUP_SEMANTICS,
295 NULL);
296 if (hFinal != INVALID_HANDLE_VALUE)
297 {
298 BY_HANDLE_FILE_INFORMATION FileData;
299 if (GetFileInformationByHandle(hFinal, &FileData))
300 {
301 Data.dwFileAttributes = FileData.dwFileAttributes;
302 Data.ftCreationTime = FileData.ftCreationTime;
303 Data.ftLastAccessTime = FileData.ftLastAccessTime;
304 Data.ftLastWriteTime = FileData.ftLastWriteTime;
305 Data.nFileSizeHigh = FileData.nFileSizeHigh;
306 Data.nFileSizeLow = FileData.nFileSizeLow;
307 }
308 CloseHandle(hFinal);
309 }
310 else if (GetLastError() != ERROR_SHARING_VIOLATION)
311 {
312 rc = RTErrConvertFromWin32(GetLastError());
313 RTUtf16Free(pwszPath);
314 return rc;
315 }
316 }
317
318 RTUtf16Free(pwszPath);
319
320 /*
321 * Setup the returned data.
322 */
323 pObjInfo->cbObject = ((uint64_t)Data.nFileSizeHigh << 32)
324 | (uint64_t)Data.nFileSizeLow;
325 pObjInfo->cbAllocated = pObjInfo->cbObject;
326
327 Assert(sizeof(uint64_t) == sizeof(Data.ftCreationTime));
328 RTTimeSpecSetNtTime(&pObjInfo->BirthTime, *(uint64_t *)&Data.ftCreationTime);
329 RTTimeSpecSetNtTime(&pObjInfo->AccessTime, *(uint64_t *)&Data.ftLastAccessTime);
330 RTTimeSpecSetNtTime(&pObjInfo->ModificationTime, *(uint64_t *)&Data.ftLastWriteTime);
331 pObjInfo->ChangeTime = pObjInfo->ModificationTime;
332
333 pObjInfo->Attr.fMode = rtFsModeFromDos((Data.dwFileAttributes << RTFS_DOS_SHIFT) & RTFS_DOS_MASK_NT,
334 pszPath, strlen(pszPath));
335
336 /*
337 * Requested attributes (we cannot provide anything actually).
338 */
339 switch (enmAdditionalAttribs)
340 {
341 case RTFSOBJATTRADD_NOTHING:
342 pObjInfo->Attr.enmAdditional = RTFSOBJATTRADD_NOTHING;
343 break;
344
345 case RTFSOBJATTRADD_UNIX:
346 pObjInfo->Attr.enmAdditional = RTFSOBJATTRADD_UNIX;
347 pObjInfo->Attr.u.Unix.uid = ~0U;
348 pObjInfo->Attr.u.Unix.gid = ~0U;
349 pObjInfo->Attr.u.Unix.cHardlinks = 1;
350 pObjInfo->Attr.u.Unix.INodeIdDevice = 0; /** @todo use volume serial number */
351 pObjInfo->Attr.u.Unix.INodeId = 0; /** @todo use fileid (see GetFileInformationByHandle). */
352 pObjInfo->Attr.u.Unix.fFlags = 0;
353 pObjInfo->Attr.u.Unix.GenerationId = 0;
354 pObjInfo->Attr.u.Unix.Device = 0;
355 break;
356
357 case RTFSOBJATTRADD_UNIX_OWNER:
358 pObjInfo->Attr.enmAdditional = RTFSOBJATTRADD_UNIX_OWNER;
359 pObjInfo->Attr.u.UnixOwner.uid = ~0U;
360 pObjInfo->Attr.u.UnixOwner.szName[0] = '\0'; /** @todo return something sensible here. */
361 break;
362
363 case RTFSOBJATTRADD_UNIX_GROUP:
364 pObjInfo->Attr.enmAdditional = RTFSOBJATTRADD_UNIX_GROUP;
365 pObjInfo->Attr.u.UnixGroup.gid = ~0U;
366 pObjInfo->Attr.u.UnixGroup.szName[0] = '\0';
367 break;
368
369 case RTFSOBJATTRADD_EASIZE:
370 pObjInfo->Attr.enmAdditional = RTFSOBJATTRADD_EASIZE;
371 pObjInfo->Attr.u.EASize.cb = 0;
372 break;
373
374 default:
375 AssertMsgFailed(("Impossible!\n"));
376 return VERR_INTERNAL_ERROR;
377 }
378
379 return VINF_SUCCESS;
380}
381
382
383RTR3DECL(int) RTPathSetTimes(const char *pszPath, PCRTTIMESPEC pAccessTime, PCRTTIMESPEC pModificationTime,
384 PCRTTIMESPEC pChangeTime, PCRTTIMESPEC pBirthTime)
385{
386 return RTPathSetTimesEx(pszPath, pAccessTime, pModificationTime, pChangeTime, pBirthTime, RTPATH_F_ON_LINK);
387}
388
389
390RTR3DECL(int) RTPathSetTimesEx(const char *pszPath, PCRTTIMESPEC pAccessTime, PCRTTIMESPEC pModificationTime,
391 PCRTTIMESPEC pChangeTime, PCRTTIMESPEC pBirthTime, uint32_t fFlags)
392{
393 /*
394 * Validate input.
395 */
396 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
397 AssertReturn(*pszPath, VERR_INVALID_PARAMETER);
398 AssertPtrNullReturn(pAccessTime, VERR_INVALID_POINTER);
399 AssertPtrNullReturn(pModificationTime, VERR_INVALID_POINTER);
400 AssertPtrNullReturn(pChangeTime, VERR_INVALID_POINTER);
401 AssertPtrNullReturn(pBirthTime, VERR_INVALID_POINTER);
402 AssertMsgReturn(RTPATH_F_IS_VALID(fFlags, 0), ("%#x\n", fFlags), VERR_INVALID_PARAMETER);
403
404 /*
405 * Convert the path.
406 */
407 PRTUTF16 pwszPath;
408 int rc = RTStrToUtf16(pszPath, &pwszPath);
409 if (RT_SUCCESS(rc))
410 {
411 HANDLE hFile;
412 if (fFlags & RTPATH_F_FOLLOW_LINK)
413 hFile = CreateFileW(pwszPath,
414 FILE_WRITE_ATTRIBUTES, /* dwDesiredAccess */
415 FILE_SHARE_WRITE | FILE_SHARE_READ | FILE_SHARE_DELETE, /* dwShareMode */
416 NULL, /* security attribs */
417 OPEN_EXISTING, /* dwCreationDisposition */
418 FILE_FLAG_BACKUP_SEMANTICS | FILE_ATTRIBUTE_NORMAL,
419 NULL);
420 else
421 {
422/** @todo Symlink: Test RTPathSetTimesEx on Windows. (The code is disabled
423 * because it's not tested yet.) */
424#if 0 //def FILE_FLAG_OPEN_REPARSE_POINT
425 hFile = CreateFileW(pwszPath,
426 FILE_WRITE_ATTRIBUTES, /* dwDesiredAccess */
427 FILE_SHARE_WRITE | FILE_SHARE_READ | FILE_SHARE_DELETE, /* dwShareMode */
428 NULL, /* security attribs */
429 OPEN_EXISTING, /* dwCreationDisposition */
430 FILE_FLAG_BACKUP_SEMANTICS | FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OPEN_REPARSE_POINT,
431 NULL);
432
433 if (hFile == INVALID_HANDLE_VALUE && GetLastError() == ERROR_INVALID_PARAMETER)
434#endif
435 hFile = CreateFileW(pwszPath,
436 FILE_WRITE_ATTRIBUTES, /* dwDesiredAccess */
437 FILE_SHARE_WRITE | FILE_SHARE_READ | FILE_SHARE_DELETE, /* dwShareMode */
438 NULL, /* security attribs */
439 OPEN_EXISTING, /* dwCreationDisposition */
440 FILE_FLAG_BACKUP_SEMANTICS | FILE_ATTRIBUTE_NORMAL,
441 NULL);
442 }
443 if (hFile != INVALID_HANDLE_VALUE)
444 {
445 /*
446 * Check if it's a no-op.
447 */
448 if (!pAccessTime && !pModificationTime && !pBirthTime)
449 rc = VINF_SUCCESS; /* NOP */
450 else
451 {
452 /*
453 * Convert the input and call the API.
454 */
455 FILETIME CreationTimeFT;
456 PFILETIME pCreationTimeFT = NULL;
457 if (pBirthTime)
458 pCreationTimeFT = RTTimeSpecGetNtFileTime(pBirthTime, &CreationTimeFT);
459
460 FILETIME LastAccessTimeFT;
461 PFILETIME pLastAccessTimeFT = NULL;
462 if (pAccessTime)
463 pLastAccessTimeFT = RTTimeSpecGetNtFileTime(pAccessTime, &LastAccessTimeFT);
464
465 FILETIME LastWriteTimeFT;
466 PFILETIME pLastWriteTimeFT = NULL;
467 if (pModificationTime)
468 pLastWriteTimeFT = RTTimeSpecGetNtFileTime(pModificationTime, &LastWriteTimeFT);
469
470 if (SetFileTime(hFile, pCreationTimeFT, pLastAccessTimeFT, pLastWriteTimeFT))
471 rc = VINF_SUCCESS;
472 else
473 {
474 DWORD Err = GetLastError();
475 rc = RTErrConvertFromWin32(Err);
476 Log(("RTPathSetTimes('%s', %p, %p, %p, %p): SetFileTime failed with lasterr %d (%Rrc)\n",
477 pszPath, pAccessTime, pModificationTime, pChangeTime, pBirthTime, Err, rc));
478 }
479 }
480 BOOL fRc = CloseHandle(hFile); Assert(fRc); NOREF(fRc);
481 }
482 else
483 {
484 DWORD Err = GetLastError();
485 rc = RTErrConvertFromWin32(Err);
486 Log(("RTPathSetTimes('%s',,,,): failed with %Rrc and lasterr=%u\n", pszPath, rc, Err));
487 }
488
489 RTUtf16Free(pwszPath);
490 }
491
492 LogFlow(("RTPathSetTimes(%p:{%s}, %p:{%RDtimespec}, %p:{%RDtimespec}, %p:{%RDtimespec}, %p:{%RDtimespec}): return %Rrc\n",
493 pszPath, pszPath, pAccessTime, pAccessTime, pModificationTime, pModificationTime,
494 pChangeTime, pChangeTime, pBirthTime, pBirthTime));
495 return rc;
496}
497
498
499
500
501/**
502 * Internal worker for RTFileRename and RTFileMove.
503 *
504 * @returns iprt status code.
505 * @param pszSrc The source filename.
506 * @param pszDst The destination filename.
507 * @param fFlags The windows MoveFileEx flags.
508 * @param fFileType The filetype. We use the RTFMODE filetypes here. If it's 0,
509 * anything goes. If it's RTFS_TYPE_DIRECTORY we'll check that the
510 * source is a directory. If Its RTFS_TYPE_FILE we'll check that it's
511 * not a directory (we are NOT checking whether it's a file).
512 */
513DECLHIDDEN(int) rtPathWin32MoveRename(const char *pszSrc, const char *pszDst, uint32_t fFlags, RTFMODE fFileType)
514{
515 /*
516 * Convert the strings.
517 */
518 PRTUTF16 pwszSrc;
519 int rc = RTStrToUtf16(pszSrc, &pwszSrc);
520 if (RT_SUCCESS(rc))
521 {
522 PRTUTF16 pwszDst;
523 rc = RTStrToUtf16(pszDst, &pwszDst);
524 if (RT_SUCCESS(rc))
525 {
526 /*
527 * Check object type if requested.
528 * This is open to race conditions.
529 */
530 if (fFileType)
531 {
532 DWORD dwAttr = GetFileAttributesW(pwszSrc);
533 if (dwAttr == INVALID_FILE_ATTRIBUTES)
534 rc = RTErrConvertFromWin32(GetLastError());
535 else if (RTFS_IS_DIRECTORY(fFileType))
536 rc = dwAttr & FILE_ATTRIBUTE_DIRECTORY ? VINF_SUCCESS : VERR_NOT_A_DIRECTORY;
537 else
538 rc = dwAttr & FILE_ATTRIBUTE_DIRECTORY ? VERR_IS_A_DIRECTORY : VINF_SUCCESS;
539 }
540 if (RT_SUCCESS(rc))
541 {
542 if (MoveFileExW(pwszSrc, pwszDst, fFlags))
543 rc = VINF_SUCCESS;
544 else
545 {
546 DWORD Err = GetLastError();
547 rc = RTErrConvertFromWin32(Err);
548 Log(("MoveFileExW('%s', '%s', %#x, %RTfmode): fails with rc=%Rrc & lasterr=%d\n",
549 pszSrc, pszDst, fFlags, fFileType, rc, Err));
550 }
551 }
552 RTUtf16Free(pwszDst);
553 }
554 RTUtf16Free(pwszSrc);
555 }
556 return rc;
557}
558
559
560RTR3DECL(int) RTPathRename(const char *pszSrc, const char *pszDst, unsigned fRename)
561{
562 /*
563 * Validate input.
564 */
565 AssertMsgReturn(VALID_PTR(pszSrc), ("%p\n", pszSrc), VERR_INVALID_POINTER);
566 AssertMsgReturn(VALID_PTR(pszDst), ("%p\n", pszDst), VERR_INVALID_POINTER);
567 AssertMsgReturn(*pszSrc, ("%p\n", pszSrc), VERR_INVALID_PARAMETER);
568 AssertMsgReturn(*pszDst, ("%p\n", pszDst), VERR_INVALID_PARAMETER);
569 AssertMsgReturn(!(fRename & ~RTPATHRENAME_FLAGS_REPLACE), ("%#x\n", fRename), VERR_INVALID_PARAMETER);
570
571 /*
572 * Call the worker.
573 */
574 int rc = rtPathWin32MoveRename(pszSrc, pszDst, fRename & RTPATHRENAME_FLAGS_REPLACE ? MOVEFILE_REPLACE_EXISTING : 0, 0);
575
576 LogFlow(("RTPathRename(%p:{%s}, %p:{%s}, %#x): returns %Rrc\n", pszSrc, pszSrc, pszDst, pszDst, fRename, rc));
577 return rc;
578}
579
580
581RTDECL(bool) RTPathExists(const char *pszPath)
582{
583 return RTPathExistsEx(pszPath, RTPATH_F_FOLLOW_LINK);
584}
585
586
587RTDECL(bool) RTPathExistsEx(const char *pszPath, uint32_t fFlags)
588{
589 /*
590 * Validate input.
591 */
592 AssertPtrReturn(pszPath, false);
593 AssertReturn(*pszPath, false);
594 Assert(RTPATH_F_IS_VALID(fFlags, 0));
595
596 /*
597 * Try query file info.
598 */
599 DWORD dwAttr;
600 PRTUTF16 pwszPath;
601 int rc = RTStrToUtf16(pszPath, &pwszPath);
602 if (RT_SUCCESS(rc))
603 {
604 dwAttr = GetFileAttributesW(pwszPath);
605 RTUtf16Free(pwszPath);
606 }
607 else
608 dwAttr = INVALID_FILE_ATTRIBUTES;
609 if (dwAttr == INVALID_FILE_ATTRIBUTES)
610 return false;
611
612#ifdef FILE_ATTRIBUTE_REPARSE_POINT
613 if ( (fFlags & RTPATH_F_FOLLOW_LINK)
614 && (dwAttr & FILE_ATTRIBUTE_REPARSE_POINT))
615 {
616 AssertFailed();
617 /** @todo Symlinks: RTPathExists+RTPathExistsEx is misbehaving on symbolic
618 * links on Windows. */
619 }
620#endif
621
622 return true;
623}
624
625
626RTDECL(int) RTPathGetCurrent(char *pszPath, size_t cchPath)
627{
628 int rc;
629
630 /*
631 * GetCurrentDirectory may in some cases omit the drive letter, according
632 * to MSDN, thus the GetFullPathName call.
633 */
634 RTUTF16 wszCurPath[RTPATH_MAX];
635 if (GetCurrentDirectoryW(RTPATH_MAX, wszCurPath))
636 {
637 RTUTF16 wszFullPath[RTPATH_MAX];
638 if (GetFullPathNameW(wszCurPath, RTPATH_MAX, wszFullPath, NULL))
639 rc = RTUtf16ToUtf8Ex(&wszFullPath[0], RTSTR_MAX, &pszPath, cchPath, NULL);
640 else
641 rc = RTErrConvertFromWin32(GetLastError());
642 }
643 else
644 rc = RTErrConvertFromWin32(GetLastError());
645 return rc;
646}
647
648
649RTDECL(int) RTPathSetCurrent(const char *pszPath)
650{
651 /*
652 * Validate input.
653 */
654 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
655 AssertReturn(*pszPath, VERR_INVALID_PARAMETER);
656
657 /*
658 * This interface is almost identical to the Windows API.
659 */
660 PRTUTF16 pwszPath;
661 int rc = RTStrToUtf16(pszPath, &pwszPath);
662 if (RT_SUCCESS(rc))
663 {
664 /** @todo improve the slash stripping a bit? */
665 size_t cwc = RTUtf16Len(pwszPath);
666 if ( cwc >= 2
667 && ( pwszPath[cwc - 1] == L'/'
668 || pwszPath[cwc - 1] == L'\\')
669 && pwszPath[cwc - 2] != ':')
670 pwszPath[cwc - 1] = L'\0';
671
672 if (!SetCurrentDirectoryW(pwszPath))
673 rc = RTErrConvertFromWin32(GetLastError());
674
675 RTUtf16Free(pwszPath);
676 }
677 return rc;
678}
679
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette