VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/posix/dir-posix.cpp@ 7689

Last change on this file since 7689 was 7418, checked in by vboxsync, 17 years ago

UCS-2 -> UTF-16.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 14.7 KB
Line 
1/* $Id: dir-posix.cpp 7418 2008-03-10 16:01:58Z vboxsync $ */
2/** @file
3 * innotek Portable Runtime - Directory manipulation, POSIX.
4 */
5
6/*
7 * Copyright (C) 2006-2007 innotek GmbH
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_DIR
32#include <errno.h>
33#include <unistd.h>
34#include <sys/types.h>
35#include <sys/stat.h>
36#include <dirent.h>
37#include <stdio.h>
38
39#include <iprt/dir.h>
40#include <iprt/path.h>
41#include <iprt/alloc.h>
42#include <iprt/alloca.h>
43#include <iprt/string.h>
44#include <iprt/assert.h>
45#include <iprt/err.h>
46#include <iprt/log.h>
47#include "internal/dir.h"
48#include "internal/fs.h"
49#include "internal/path.h"
50
51#if !defined(RT_OS_SOLARIS)
52# define HAVE_DIRENT_D_TYPE 1
53#endif
54
55
56RTDECL(bool) RTDirExists(const char *pszPath)
57{
58 bool fRc = false;
59 char *pszNativePath;
60 int rc = rtPathToNative(&pszNativePath, pszPath);
61 if (RT_SUCCESS(rc))
62 {
63 struct stat s;
64 fRc = !stat(pszNativePath, &s)
65 && S_ISDIR(s.st_mode);
66
67 rtPathFreeNative(pszNativePath);
68 }
69
70 LogFlow(("RTDirExists(%p={%s}): returns %RTbool\n", pszPath, pszPath, fRc));
71 return fRc;
72}
73
74
75RTDECL(int) RTDirCreate(const char *pszPath, RTFMODE fMode)
76{
77 int rc;
78 fMode = rtFsModeNormalize(fMode, pszPath, 0);
79 if (rtFsModeIsValidPermissions(fMode))
80 {
81 char *pszNativePath;
82 rc = rtPathToNative(&pszNativePath, pszPath);
83 if (RT_SUCCESS(rc))
84 {
85 if (mkdir(pszNativePath, fMode & RTFS_UNIX_MASK))
86 {
87#ifdef RT_OS_SOLARIS
88 if (errno == ENOSYS) /* ENOSYS has a slight different meaning (mkdir on nfs mount point). */
89 rc = VERR_ALREADY_EXISTS;
90 else
91#endif
92 rc = RTErrConvertFromErrno(errno);
93 }
94 }
95
96 rtPathFreeNative(pszNativePath);
97 }
98 else
99 {
100 AssertMsgFailed(("Invalid file mode! %RTfmode\n", fMode));
101 rc = VERR_INVALID_FMODE;
102 }
103 LogFlow(("RTDirCreate(%p={%s}, %RTfmode): returns %Rrc\n", pszPath, pszPath, fMode, rc));
104 return rc;
105}
106
107
108RTDECL(int) RTDirRemove(const char *pszPath)
109{
110 char *pszNativePath;
111 int rc = rtPathToNative(&pszNativePath, pszPath);
112 if (RT_SUCCESS(rc))
113 {
114 if (rmdir(pszNativePath))
115 rc = RTErrConvertFromErrno(errno);
116
117 rtPathFreeNative(pszNativePath);
118 }
119
120 LogFlow(("RTDirRemove(%p={%s}): returns %Rrc\n", pszPath, pszPath, rc));
121 return rc;
122}
123
124
125int rtOpenDirNative(PRTDIR pDir, char *pszPathBuf)
126{
127 /*
128 * Convert to a native path and try opendir.
129 */
130 char *pszNativePath;
131 int rc = rtPathToNative(&pszNativePath, pDir->pszPath);
132 if (RT_SUCCESS(rc))
133 {
134 pDir->pDir = opendir(pszNativePath);
135 if (pDir->pDir)
136 {
137 /*
138 * Init data.
139 */
140 pDir->fDataUnread = false;
141 memset(&pDir->Data, 0, sizeof(pDir->Data)); /* not strictly necessary */
142 }
143 else
144 rc = RTErrConvertFromErrno(errno);
145
146 rtPathFreeNative(pszNativePath);
147 }
148
149 return rc;
150}
151
152
153RTDECL(int) RTDirClose(PRTDIR pDir)
154{
155 /*
156 * Validate input.
157 */
158 if (!pDir)
159 return VERR_INVALID_PARAMETER;
160 if (pDir->u32Magic != RTDIR_MAGIC)
161 {
162 AssertMsgFailed(("Invalid pDir=%p\n", pDir));
163 return VERR_INVALID_PARAMETER;
164 }
165
166 /*
167 * Close the handle.
168 */
169 int rc = VINF_SUCCESS;
170 pDir->u32Magic = RTDIR_MAGIC_DEAD;
171 if (closedir(pDir->pDir))
172 {
173 rc = RTErrConvertFromErrno(errno);
174 AssertMsgFailed(("closedir(%p) -> errno=%d (%Rrc)\n", pDir->pDir, errno, rc));
175 }
176
177 RTMemFree(pDir);
178 return rc;
179}
180
181
182/**
183 * Ensure that there is unread data in the buffer
184 * and that there is a converted filename hanging around.
185 *
186 * @returns IPRT status code.
187 * @param pDir the open directory. Fully validated.
188 */
189static int rtDirReadMore(PRTDIR pDir)
190{
191 /** @todo try avoid the rematching on buffer overflow errors. */
192 for (;;)
193 {
194 /*
195 * Fetch data?
196 */
197 if (!pDir->fDataUnread)
198 {
199 struct dirent *pResult = NULL;
200 int rc = readdir_r(pDir->pDir, &pDir->Data, &pResult);
201 if (rc)
202 {
203 rc = RTErrConvertFromErrno(rc);
204 AssertRC(rc);
205 return rc;
206 }
207 if (!pResult)
208 return VERR_NO_MORE_FILES;
209 }
210
211#ifndef RT_DONT_CONVERT_FILENAMES
212 /*
213 * Convert the filename to UTF-8.
214 */
215 if (!pDir->pszName)
216 {
217 int rc = rtPathFromNativeEx(&pDir->pszName, pDir->Data.d_name, pDir->pszPath);
218 if (RT_FAILURE(rc))
219 {
220 pDir->pszName = NULL;
221 return rc;
222 }
223 pDir->cchName = strlen(pDir->pszName);
224 }
225 if ( !pDir->pfnFilter
226 || pDir->pfnFilter(pDir, pDir->pszName))
227 break;
228 RTStrFree(pDir->pszName);
229 pDir->pszName = NULL;
230#else
231 if ( !pDir->pfnFilter
232 || pDir->pfnFilter(pDir, pDir->Data.d_name))
233 break;
234#endif
235 pDir->fDataUnread = false;
236 }
237
238 pDir->fDataUnread = true;
239 return VINF_SUCCESS;
240}
241
242
243#ifdef HAVE_DIRENT_D_TYPE
244/**
245 * Converts the d_type field to IPRT directory entry type.
246 *
247 * @returns IPRT directory entry type.
248 * @param Unix
249 */
250static RTDIRENTRYTYPE rtDirType(int iType)
251{
252 switch (iType)
253 {
254 case DT_UNKNOWN: return RTDIRENTRYTYPE_UNKNOWN;
255 case DT_FIFO: return RTDIRENTRYTYPE_FIFO;
256 case DT_CHR: return RTDIRENTRYTYPE_DEV_CHAR;
257 case DT_DIR: return RTDIRENTRYTYPE_DIRECTORY;
258 case DT_BLK: return RTDIRENTRYTYPE_DEV_BLOCK;
259 case DT_REG: return RTDIRENTRYTYPE_FILE;
260 case DT_LNK: return RTDIRENTRYTYPE_SYMLINK;
261 case DT_SOCK: return RTDIRENTRYTYPE_SOCKET;
262 case DT_WHT: return RTDIRENTRYTYPE_WHITEOUT;
263 default:
264 AssertMsgFailed(("iType=%d\n", iType));
265 return RTDIRENTRYTYPE_UNKNOWN;
266 }
267}
268#endif /*HAVE_DIRENT_D_TYPE */
269
270
271RTDECL(int) RTDirRead(PRTDIR pDir, PRTDIRENTRY pDirEntry, unsigned *pcbDirEntry)
272{
273 /*
274 * Validate and digest input.
275 */
276 if (!rtDirValidHandle(pDir))
277 return VERR_INVALID_PARAMETER;
278 AssertMsgReturn(VALID_PTR(pDirEntry), ("%p\n", pDirEntry), VERR_INVALID_POINTER);
279
280 unsigned cbDirEntry = sizeof(*pDirEntry);
281 if (pcbDirEntry)
282 {
283 AssertMsgReturn(VALID_PTR(pcbDirEntry), ("%p\n", pcbDirEntry), VERR_INVALID_POINTER);
284 cbDirEntry = *pcbDirEntry;
285 AssertMsgReturn(cbDirEntry >= (unsigned)RT_OFFSETOF(RTDIRENTRY, szName[2]),
286 ("Invalid *pcbDirEntry=%d (min %d)\n", *pcbDirEntry, RT_OFFSETOF(RTDIRENTRYEX, szName[2])),
287 VERR_INVALID_PARAMETER);
288 }
289
290 /*
291 * Fetch more data if necessary and/or convert the name.
292 */
293 int rc = rtDirReadMore(pDir);
294 if (RT_SUCCESS(rc))
295 {
296 /*
297 * Check if we've got enough space to return the data.
298 */
299#ifdef RT_DONT_CONVERT_FILENAMES
300 const char *pszName = pDir->Data.d_name;
301 const size_t cchName = strlen(pszName);
302#else
303 const char *pszName = pDir->pszName;
304 const size_t cchName = pDir->cchName;
305#endif
306 const size_t cbRequired = RT_OFFSETOF(RTDIRENTRY, szName[1]) + cchName;
307 if (pcbDirEntry)
308 *pcbDirEntry = cbRequired;
309 if (cbRequired <= cbDirEntry)
310 {
311 /*
312 * Setup the returned data.
313 */
314 pDirEntry->INodeId = pDir->Data.d_ino; /* may need #ifdefing later */
315#ifdef HAVE_DIRENT_D_TYPE
316 pDirEntry->enmType = rtDirType(pDir->Data.d_type);
317#else
318 pDirEntry->enmType = RTDIRENTRYTYPE_UNKNOWN;
319#endif
320 pDirEntry->cbName = (uint16_t)cchName;
321 Assert(pDirEntry->cbName == cchName);
322 memcpy(pDirEntry->szName, pszName, cchName + 1);
323
324 /* free cached data */
325 pDir->fDataUnread = false;
326#ifndef RT_DONT_CONVERT_FILENAMES
327 RTStrFree(pDir->pszName);
328 pDir->pszName = NULL;
329#endif
330 }
331 else
332 rc = VERR_BUFFER_OVERFLOW;
333 }
334
335 LogFlow(("RTDirRead(%p:{%s}, %p:{%s}, %p:{%u}): returns %Rrc\n",
336 pDir, pDir->pszPath, pDirEntry, RT_SUCCESS(rc) ? pDirEntry->szName : "<failed>",
337 pcbDirEntry, pcbDirEntry ? *pcbDirEntry : 0, rc));
338 return rc;
339}
340
341
342/**
343 * Fills dummy info into the info structure.
344 * This function is called if we cannot stat the file.
345 *
346 * @param pInfo The struct in question.
347 * @param
348 */
349static void rtDirSetDummyInfo(PRTFSOBJINFO pInfo, RTDIRENTRYTYPE enmType)
350{
351 pInfo->cbObject = 0;
352 pInfo->cbAllocated = 0;
353 RTTimeSpecSetNano(&pInfo->AccessTime, 0);
354 RTTimeSpecSetNano(&pInfo->ModificationTime, 0);
355 RTTimeSpecSetNano(&pInfo->ChangeTime, 0);
356 RTTimeSpecSetNano(&pInfo->BirthTime, 0);
357 memset(&pInfo->Attr, 0, sizeof(pInfo->Attr));
358 pInfo->Attr.enmAdditional = RTFSOBJATTRADD_NOTHING;
359 switch (enmType)
360 {
361 default:
362 case RTDIRENTRYTYPE_UNKNOWN: pInfo->Attr.fMode = RTFS_DOS_NT_NORMAL;
363 case RTDIRENTRYTYPE_FIFO: pInfo->Attr.fMode = RTFS_DOS_NT_NORMAL | RTFS_TYPE_FIFO;
364 case RTDIRENTRYTYPE_DEV_CHAR: pInfo->Attr.fMode = RTFS_DOS_NT_NORMAL | RTFS_TYPE_DEV_CHAR;
365 case RTDIRENTRYTYPE_DIRECTORY: pInfo->Attr.fMode = RTFS_DOS_DIRECTORY | RTFS_TYPE_DIRECTORY;
366 case RTDIRENTRYTYPE_DEV_BLOCK: pInfo->Attr.fMode = RTFS_DOS_NT_NORMAL | RTFS_TYPE_DEV_BLOCK;
367 case RTDIRENTRYTYPE_FILE: pInfo->Attr.fMode = RTFS_DOS_NT_NORMAL | RTFS_TYPE_FILE;
368 case RTDIRENTRYTYPE_SYMLINK: pInfo->Attr.fMode = RTFS_DOS_NT_NORMAL | RTFS_TYPE_SYMLINK;
369 case RTDIRENTRYTYPE_SOCKET: pInfo->Attr.fMode = RTFS_DOS_NT_NORMAL | RTFS_TYPE_SOCKET;
370 case RTDIRENTRYTYPE_WHITEOUT: pInfo->Attr.fMode = RTFS_DOS_NT_NORMAL | RTFS_TYPE_WHITEOUT;
371 }
372}
373
374
375RTDECL(int) RTDirReadEx(PRTDIR pDir, PRTDIRENTRYEX pDirEntry, unsigned *pcbDirEntry, RTFSOBJATTRADD enmAdditionalAttribs)
376{
377 /*
378 * Validate and digest input.
379 */
380 if (!rtDirValidHandle(pDir))
381 return VERR_INVALID_PARAMETER;
382 AssertMsgReturn(VALID_PTR(pDirEntry), ("%p\n", pDirEntry), VERR_INVALID_POINTER);
383 AssertMsgReturn( enmAdditionalAttribs >= RTFSOBJATTRADD_NOTHING
384 && enmAdditionalAttribs <= RTFSOBJATTRADD_LAST,
385 ("Invalid enmAdditionalAttribs=%p\n", enmAdditionalAttribs),
386 VERR_INVALID_PARAMETER);
387 unsigned cbDirEntry = sizeof(*pDirEntry);
388 if (pcbDirEntry)
389 {
390 AssertMsgReturn(VALID_PTR(pcbDirEntry), ("%p\n", pcbDirEntry), VERR_INVALID_POINTER);
391 cbDirEntry = *pcbDirEntry;
392 AssertMsgReturn(cbDirEntry >= (unsigned)RT_OFFSETOF(RTDIRENTRYEX, szName[2]),
393 ("Invalid *pcbDirEntry=%d (min %d)\n", *pcbDirEntry, RT_OFFSETOF(RTDIRENTRYEX, szName[2])),
394 VERR_INVALID_PARAMETER);
395 }
396
397 /*
398 * Fetch more data if necessary and/or convert the name.
399 */
400 int rc = rtDirReadMore(pDir);
401 if (RT_SUCCESS(rc))
402 {
403 /*
404 * Check if we've got enough space to return the data.
405 */
406#ifdef RT_DONT_CONVERT_FILENAMES
407 const char *pszName = pDir->Data.d_name;
408 const size_t cchName = strlen(pszName);
409#else
410 const char *pszName = pDir->pszName;
411 const size_t cchName = pDir->cchName;
412#endif
413 const size_t cbRequired = RT_OFFSETOF(RTDIRENTRYEX, szName[1]) + cchName;
414 if (pcbDirEntry)
415 *pcbDirEntry = cbRequired;
416 if (cbRequired <= cbDirEntry)
417 {
418 /*
419 * Setup the returned data.
420 */
421 pDirEntry->cwcShortName = 0;
422 pDirEntry->wszShortName[0] = 0;
423 pDirEntry->cbName = (uint16_t)cchName;
424 Assert(pDirEntry->cbName == cchName);
425 memcpy(pDirEntry->szName, pszName, cchName + 1);
426
427 /* get the info data */
428 size_t cch = cchName + pDir->cchPath + 1;
429 char *pszNamePath = (char *)alloca(cch);
430 if (pszNamePath)
431 {
432 memcpy(pszNamePath, pDir->pszPath, pDir->cchPath);
433 memcpy(pszNamePath + pDir->cchPath, pszName, cchName + 1);
434 rc = RTPathQueryInfo(pszNamePath, &pDirEntry->Info, enmAdditionalAttribs);
435 }
436 else
437 rc = VERR_NO_MEMORY;
438 if (RT_FAILURE(rc))
439 {
440#ifdef HAVE_DIRENT_D_TYPE
441 rtDirSetDummyInfo(&pDirEntry->Info, rtDirType(pDir->Data.d_type));
442#else
443 rtDirSetDummyInfo(&pDirEntry->Info, RTDIRENTRYTYPE_UNKNOWN);
444#endif
445 rc = VWRN_NO_DIRENT_INFO;
446 }
447
448 /* free cached data */
449 pDir->fDataUnread = false;
450#ifndef RT_DONT_CONVERT_FILENAMES
451 RTStrFree(pDir->pszName);
452 pDir->pszName = NULL;
453#endif
454 }
455 else
456 rc = VERR_BUFFER_OVERFLOW;
457 }
458
459 return rc;
460}
461
462
463RTDECL(int) RTDirRename(const char *pszSrc, const char *pszDst, unsigned fRename)
464{
465 /*
466 * Validate input.
467 */
468 AssertMsgReturn(VALID_PTR(pszSrc), ("%p\n", pszSrc), VERR_INVALID_POINTER);
469 AssertMsgReturn(VALID_PTR(pszDst), ("%p\n", pszDst), VERR_INVALID_POINTER);
470 AssertMsgReturn(*pszSrc, ("%p\n", pszSrc), VERR_INVALID_PARAMETER);
471 AssertMsgReturn(*pszDst, ("%p\n", pszDst), VERR_INVALID_PARAMETER);
472 AssertMsgReturn(!(fRename & ~RTPATHRENAME_FLAGS_REPLACE), ("%#x\n", fRename), VERR_INVALID_PARAMETER);
473
474 /*
475 * Take common cause with RTPathRename.
476 */
477 int rc = rtPathPosixRename(pszSrc, pszDst, fRename, RTFS_TYPE_DIRECTORY);
478
479 LogFlow(("RTDirRename(%p:{%s}, %p:{%s}): returns %Rrc\n",
480 pszSrc, pszSrc, pszDst, pszDst, rc));
481 return rc;
482}
483
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