VirtualBox

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

Last change on this file since 1507 was 1, checked in by vboxsync, 55 years ago

import

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