VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/win/dir-win.cpp@ 25640

Last change on this file since 25640 was 25292, checked in by vboxsync, 15 years ago

RTDirReadEx parameter to resolve symlinks.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 15.1 KB
Line 
1/* $Id: dir-win.cpp 25292 2009-12-10 10:29:57Z vboxsync $ */
2/** @file
3 * IPRT - Directory, win32.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
27 * Clara, CA 95054 USA or visit http://www.sun.com if you need
28 * additional information or have any questions.
29 */
30
31
32/*******************************************************************************
33* Header Files *
34*******************************************************************************/
35#define LOG_GROUP RTLOGGROUP_DIR
36#include <Windows.h>
37#include <io.h>
38
39#include <iprt/dir.h>
40#include <iprt/path.h>
41#include <iprt/alloc.h>
42#include <iprt/string.h>
43#include <iprt/assert.h>
44#include <iprt/param.h>
45#include <iprt/err.h>
46#include <iprt/file.h>
47#include <iprt/log.h>
48#include "internal/fs.h"
49#include "internal/path.h"
50#include "internal/dir.h"
51
52
53
54RTDECL(int) RTDirCreate(const char *pszPath, RTFMODE fMode)
55{
56 /*
57 * Validate the file mode.
58 */
59 int rc;
60 fMode = rtFsModeNormalize(fMode, pszPath, 0);
61 if (rtFsModeIsValidPermissions(fMode))
62 {
63 /*
64 * Convert to UTF-16.
65 */
66 PRTUTF16 pwszString;
67 rc = RTStrToUtf16(pszPath, &pwszString);
68 AssertRC(rc);
69 if (RT_SUCCESS(rc))
70 {
71 /*
72 * Create the directory.
73 */
74 if (CreateDirectoryW((LPCWSTR)pwszString, NULL))
75 rc = VINF_SUCCESS;
76 else
77 rc = RTErrConvertFromWin32(GetLastError());
78
79 /*
80 * Turn off indexing of directory through Windows Indexing Service
81 */
82 if (RT_SUCCESS(rc))
83 {
84 if (SetFileAttributesW((LPCWSTR)pwszString, FILE_ATTRIBUTE_NOT_CONTENT_INDEXED))
85 rc = VINF_SUCCESS;
86 else
87 rc = RTErrConvertFromWin32(GetLastError());
88 }
89
90 RTUtf16Free(pwszString);
91 }
92 }
93 else
94 {
95 AssertMsgFailed(("Invalid file mode! %RTfmode\n", fMode));
96 rc = VERR_INVALID_FMODE;
97 }
98
99 LogFlow(("RTDirCreate(%p:{%s}, %RTfmode): returns %Rrc\n", pszPath, pszPath, fMode, rc));
100 return rc;
101}
102
103
104RTDECL(int) RTDirRemove(const char *pszPath)
105{
106 /*
107 * Convert to UTF-16.
108 */
109 PRTUTF16 pwszString;
110 int rc = RTStrToUtf16(pszPath, &pwszString);
111 AssertRC(rc);
112 if (RT_SUCCESS(rc))
113 {
114 /*
115 * Remove the directory.
116 */
117 if (RemoveDirectoryW((LPCWSTR)pwszString))
118 rc = VINF_SUCCESS;
119 else
120 rc = RTErrConvertFromWin32(GetLastError());
121
122 RTUtf16Free(pwszString);
123 }
124
125 LogFlow(("RTDirRemove(%p:{%s}): returns %Rrc\n", pszPath, pszPath, rc));
126 return rc;
127}
128
129
130int rtOpenDirNative(PRTDIR pDir, char *pszPathBuf)
131{
132 /*
133 * Setup the search expression.
134 *
135 * pszPathBuf is pointing to the return 4K return buffer for the RTPathReal()
136 * call in rtDirOpenCommon(), so all we gota do is check that we don't overflow
137 * it when adding the wildcard expression.
138 */
139 size_t cchExpr;
140 const char *pszExpr;
141 if (pDir->enmFilter == RTDIRFILTER_WINNT)
142 {
143 pszExpr = pDir->pszFilter;
144 cchExpr = pDir->cchFilter + 1;
145 }
146 else
147 {
148 pszExpr = "*";
149 cchExpr = sizeof("*");
150 }
151 if (pDir->cchPath + cchExpr > RTPATH_MAX)
152 return VERR_FILENAME_TOO_LONG;
153 memcpy(pszPathBuf + pDir->cchPath, pszExpr, cchExpr);
154
155
156 /*
157 * Attempt opening the search.
158 */
159 int rc = VINF_SUCCESS;
160#ifndef RT_DONT_CONVERT_FILENAMES
161 PRTUTF16 pwszName;
162 rc = RTStrToUtf16(pszPathBuf, &pwszName);
163 if (RT_SUCCESS(rc))
164 {
165 pDir->hDir = FindFirstFileW((LPCWSTR)pwszName, &pDir->Data);
166#else
167 pDir->hDir = FindFirstFileA(pszPathBuf, &pDir->Data);
168#endif
169 if (pDir->hDir != INVALID_HANDLE_VALUE)
170 pDir->fDataUnread = true;
171 /* theoretical case of an empty directory. */
172 else if (GetLastError() == ERROR_NO_MORE_FILES)
173 pDir->fDataUnread = false;
174 else
175 rc = RTErrConvertFromWin32(GetLastError());
176#ifndef RT_DONT_CONVERT_FILENAMES
177 RTUtf16Free(pwszName);
178 }
179#endif
180
181 return rc;
182}
183
184
185RTDECL(int) RTDirClose(PRTDIR pDir)
186{
187 /*
188 * Validate input.
189 */
190 if (!pDir)
191 return VERR_INVALID_PARAMETER;
192 if (pDir->u32Magic != RTDIR_MAGIC)
193 {
194 AssertMsgFailed(("Invalid pDir=%p\n", pDir));
195 return VERR_INVALID_PARAMETER;
196 }
197
198 /*
199 * Close the handle.
200 */
201 pDir->u32Magic++;
202 if (pDir->hDir != INVALID_HANDLE_VALUE)
203 {
204 BOOL fRc = FindClose(pDir->hDir);
205 Assert(fRc);
206 pDir->hDir = INVALID_HANDLE_VALUE;
207 }
208 RTStrFree(pDir->pszName);
209 pDir->pszName = NULL;
210 RTMemFree(pDir);
211
212 return VINF_SUCCESS;
213}
214
215
216RTDECL(int) RTDirRead(PRTDIR pDir, PRTDIRENTRY pDirEntry, size_t *pcbDirEntry)
217{
218 /*
219 * Validate input.
220 */
221 if (!pDir || pDir->u32Magic != RTDIR_MAGIC)
222 {
223 AssertMsgFailed(("Invalid pDir=%p\n", pDir));
224 return VERR_INVALID_PARAMETER;
225 }
226 if (!pDirEntry)
227 {
228 AssertMsgFailed(("Invalid pDirEntry=%p\n", pDirEntry));
229 return VERR_INVALID_PARAMETER;
230 }
231 size_t cbDirEntry = sizeof(*pDirEntry);
232 if (pcbDirEntry)
233 {
234 cbDirEntry = *pcbDirEntry;
235 if (cbDirEntry < RT_UOFFSETOF(RTDIRENTRY, szName[2]))
236 {
237 AssertMsgFailed(("Invalid *pcbDirEntry=%d (min %d)\n", *pcbDirEntry, RT_OFFSETOF(RTDIRENTRY, szName[2])));
238 return VERR_INVALID_PARAMETER;
239 }
240 }
241
242 /*
243 * Fetch data?
244 */
245 if (!pDir->fDataUnread)
246 {
247#ifdef RT_DONT_CONVERT_FILENAMES
248 BOOL fRc = FindNextFileA(pDir->hDir, &pDir->Data);
249
250#else
251 RTStrFree(pDir->pszName);
252 pDir->pszName = NULL;
253
254 BOOL fRc = FindNextFileW(pDir->hDir, &pDir->Data);
255#endif
256 if (!fRc)
257 {
258 int iErr = GetLastError();
259 if (pDir->hDir == INVALID_HANDLE_VALUE || iErr == ERROR_NO_MORE_FILES)
260 return VERR_NO_MORE_FILES;
261 return RTErrConvertFromWin32(iErr);
262 }
263 }
264
265#ifndef RT_DONT_CONVERT_FILENAMES
266 /*
267 * Convert the filename to UTF-8.
268 */
269 if (!pDir->pszName)
270 {
271 int rc = RTUtf16ToUtf8((PCRTUTF16)pDir->Data.cFileName, &pDir->pszName);
272 if (RT_FAILURE(rc))
273 {
274 pDir->pszName = NULL;
275 return rc;
276 }
277 pDir->cchName = strlen(pDir->pszName);
278 }
279#endif
280
281 /*
282 * Check if we've got enough space to return the data.
283 */
284#ifdef RT_DONT_CONVERT_FILENAMES
285 const char *pszName = pDir->Data.cName;
286 const size_t cchName = strlen(pszName);
287#else
288 const char *pszName = pDir->pszName;
289 const size_t cchName = pDir->cchName;
290#endif
291 const size_t cbRequired = RT_OFFSETOF(RTDIRENTRY, szName[1]) + cchName;
292 if (pcbDirEntry)
293 *pcbDirEntry = cbRequired;
294 if (cbRequired > cbDirEntry)
295 return VERR_BUFFER_OVERFLOW;
296
297 /*
298 * Setup the returned data.
299 */
300 pDir->fDataUnread = false;
301 pDirEntry->INodeId = 0; /** @todo we can use the fileid here if we must (see GetFileInformationByHandle). */
302 pDirEntry->enmType = pDir->Data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY
303 ? RTDIRENTRYTYPE_DIRECTORY : RTDIRENTRYTYPE_FILE;
304 pDirEntry->cbName = (uint16_t)cchName;
305 Assert(pDirEntry->cbName == cchName);
306 memcpy(pDirEntry->szName, pszName, cchName + 1);
307
308 return VINF_SUCCESS;
309}
310
311
312RTDECL(int) RTDirReadEx(PRTDIR pDir, PRTDIRENTRYEX pDirEntry, size_t *pcbDirEntry, RTFSOBJATTRADD enmAdditionalAttribs, uint32_t fFlags)
313{
314 /** @todo Symlinks: Find[First|Next]FileW will return info about
315 the link, so RTPATH_F_FOLLOW_LINK is not handled correctly. */
316 /*
317 * Validate input.
318 */
319 if (!pDir || pDir->u32Magic != RTDIR_MAGIC)
320 {
321 AssertMsgFailed(("Invalid pDir=%p\n", pDir));
322 return VERR_INVALID_PARAMETER;
323 }
324 if (!pDirEntry)
325 {
326 AssertMsgFailed(("Invalid pDirEntry=%p\n", pDirEntry));
327 return VERR_INVALID_PARAMETER;
328 }
329 if ( enmAdditionalAttribs < RTFSOBJATTRADD_NOTHING
330 || enmAdditionalAttribs > RTFSOBJATTRADD_LAST)
331 {
332 AssertMsgFailed(("Invalid enmAdditionalAttribs=%p\n", enmAdditionalAttribs));
333 return VERR_INVALID_PARAMETER;
334 }
335 AssertMsgReturn(RTPATH_F_IS_VALID(fFlags, 0), ("%#x\n", fFlags), VERR_INVALID_PARAMETER);
336 size_t cbDirEntry = sizeof(*pDirEntry);
337 if (pcbDirEntry)
338 {
339 cbDirEntry = *pcbDirEntry;
340 if (cbDirEntry < RT_UOFFSETOF(RTDIRENTRYEX, szName[2]))
341 {
342 AssertMsgFailed(("Invalid *pcbDirEntry=%d (min %d)\n", *pcbDirEntry, RT_OFFSETOF(RTDIRENTRYEX, szName[2])));
343 return VERR_INVALID_PARAMETER;
344 }
345 }
346
347 /*
348 * Fetch data?
349 */
350 if (!pDir->fDataUnread)
351 {
352#ifdef RT_DONT_CONVERT_FILENAMES
353 BOOL fRc = FindNextFileA(pDir->hDir, &pDir->Data);
354
355#else
356 RTStrFree(pDir->pszName);
357 pDir->pszName = NULL;
358
359 BOOL fRc = FindNextFileW(pDir->hDir, &pDir->Data);
360#endif
361 if (!fRc)
362 {
363 int iErr = GetLastError();
364 if (pDir->hDir == INVALID_HANDLE_VALUE || iErr == ERROR_NO_MORE_FILES)
365 return VERR_NO_MORE_FILES;
366 return RTErrConvertFromWin32(iErr);
367 }
368 }
369
370#ifndef RT_DONT_CONVERT_FILENAMES
371 /*
372 * Convert the filename to UTF-8.
373 */
374 if (!pDir->pszName)
375 {
376 int rc = RTUtf16ToUtf8((PCRTUTF16)pDir->Data.cFileName, &pDir->pszName);
377 if (RT_FAILURE(rc))
378 {
379 pDir->pszName = NULL;
380 return rc;
381 }
382 pDir->cchName = strlen(pDir->pszName);
383 }
384#endif
385
386 /*
387 * Check if we've got enough space to return the data.
388 */
389#ifdef RT_DONT_CONVERT_FILENAMES
390 const char *pszName = pDir->Data.cName;
391 const size_t cchName = strlen(pszName);
392#else
393 const char *pszName = pDir->pszName;
394 const size_t cchName = pDir->cchName;
395#endif
396 const size_t cbRequired = RT_OFFSETOF(RTDIRENTRYEX, szName[1]) + cchName;
397 if (pcbDirEntry)
398 *pcbDirEntry = cbRequired;
399 if (cbRequired > cbDirEntry)
400 return VERR_BUFFER_OVERFLOW;
401
402 /*
403 * Setup the returned data.
404 */
405 pDir->fDataUnread = false;
406 pDirEntry->cbName = (uint16_t)cchName;
407 Assert(pDirEntry->cbName == cchName);
408 memcpy(pDirEntry->szName, pszName, cchName + 1);
409#ifndef RT_DONT_CONVERT_FILENAMES /* this ain't nice since the whole point of this define is not to drag in conversion... */
410 if (pDir->Data.cAlternateFileName[0])
411 {
412 /* copy and calc length */
413 PCRTUTF16 pwszSrc = (PCRTUTF16)pDir->Data.cAlternateFileName;
414 PRTUTF16 pwszDst = pDirEntry->wszShortName;
415 while (*pwszSrc)
416 *pwszDst++ = *pwszSrc++;
417 pDirEntry->cwcShortName = pwszDst - &pDirEntry->wszShortName[0];
418 /* zero the rest */
419 const PRTUTF16 pwszEnd = &pDirEntry->wszShortName[RT_ELEMENTS(pDirEntry->wszShortName)];
420 while (pwszDst < pwszEnd)
421 *pwszDst++ = '\0';
422 }
423 else
424#endif
425 {
426 memset(pDirEntry->wszShortName, 0, sizeof(pDirEntry->wszShortName));
427 pDirEntry->cwcShortName = 0;
428 }
429
430 pDirEntry->Info.cbObject = ((uint64_t)pDir->Data.nFileSizeHigh << 32)
431 | (uint64_t)pDir->Data.nFileSizeLow;
432 pDirEntry->Info.cbAllocated = pDirEntry->Info.cbObject;
433
434 Assert(sizeof(uint64_t) == sizeof(pDir->Data.ftCreationTime));
435 RTTimeSpecSetNtTime(&pDirEntry->Info.BirthTime, *(uint64_t *)&pDir->Data.ftCreationTime);
436 RTTimeSpecSetNtTime(&pDirEntry->Info.AccessTime, *(uint64_t *)&pDir->Data.ftLastAccessTime);
437 RTTimeSpecSetNtTime(&pDirEntry->Info.ModificationTime, *(uint64_t *)&pDir->Data.ftLastWriteTime);
438 pDirEntry->Info.ChangeTime = pDirEntry->Info.ModificationTime;
439
440 pDirEntry->Info.Attr.fMode = rtFsModeFromDos((pDir->Data.dwFileAttributes << RTFS_DOS_SHIFT) & RTFS_DOS_MASK_NT,
441 pszName, cchName);
442
443 /*
444 * Requested attributes (we cannot provide anything actually).
445 */
446 switch (enmAdditionalAttribs)
447 {
448 case RTFSOBJATTRADD_EASIZE:
449 pDirEntry->Info.Attr.enmAdditional = RTFSOBJATTRADD_EASIZE;
450 pDirEntry->Info.Attr.u.EASize.cb = 0;
451 break;
452
453 case RTFSOBJATTRADD_UNIX:
454 pDirEntry->Info.Attr.enmAdditional = RTFSOBJATTRADD_UNIX;
455 pDirEntry->Info.Attr.u.Unix.uid = ~0U;
456 pDirEntry->Info.Attr.u.Unix.gid = ~0U;
457 pDirEntry->Info.Attr.u.Unix.cHardlinks = 1;
458 pDirEntry->Info.Attr.u.Unix.INodeIdDevice = 0; /** @todo Use the volume serial number (see GetFileInformationByHandle). */
459 pDirEntry->Info.Attr.u.Unix.INodeId = 0; /** @todo Use the fileid (see GetFileInformationByHandle). */
460 pDirEntry->Info.Attr.u.Unix.fFlags = 0;
461 pDirEntry->Info.Attr.u.Unix.GenerationId = 0;
462 pDirEntry->Info.Attr.u.Unix.Device = 0;
463 break;
464
465 case RTFSOBJATTRADD_NOTHING:
466 pDirEntry->Info.Attr.enmAdditional = RTFSOBJATTRADD_NOTHING;
467 break;
468
469 default:
470 AssertMsgFailed(("Impossible!\n"));
471 return VERR_INTERNAL_ERROR;
472 }
473
474 return VINF_SUCCESS;
475}
476
477
478RTDECL(int) RTDirRename(const char *pszSrc, const char *pszDst, unsigned fRename)
479{
480 /*
481 * Validate input.
482 */
483 AssertMsgReturn(VALID_PTR(pszSrc), ("%p\n", pszSrc), VERR_INVALID_POINTER);
484 AssertMsgReturn(VALID_PTR(pszDst), ("%p\n", pszDst), VERR_INVALID_POINTER);
485 AssertMsgReturn(*pszSrc, ("%p\n", pszSrc), VERR_INVALID_PARAMETER);
486 AssertMsgReturn(*pszDst, ("%p\n", pszDst), VERR_INVALID_PARAMETER);
487 AssertMsgReturn(!(fRename & ~RTPATHRENAME_FLAGS_REPLACE), ("%#x\n", fRename), VERR_INVALID_PARAMETER);
488
489 /*
490 * Call the worker.
491 */
492 int rc = rtPathWin32MoveRename(pszSrc, pszDst,
493 fRename & RTPATHRENAME_FLAGS_REPLACE ? MOVEFILE_REPLACE_EXISTING : 0,
494 RTFS_TYPE_DIRECTORY);
495
496 LogFlow(("RTDirRename(%p:{%s}, %p:{%s}, %#x): returns %Rrc\n", pszSrc, pszSrc, pszDst, pszDst, fRename, rc));
497 return rc;
498}
499
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