VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/posix/path-posix.cpp@ 2034

Last change on this file since 2034 was 937, checked in by vboxsync, 18 years ago

RTPathExists.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 27.5 KB
Line 
1/* $Id: path-posix.cpp 937 2007-02-15 20:59:20Z vboxsync $ */
2/** @file
3 * InnoTek Portable Runtime - Path 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_PATH
27#include <stdlib.h>
28#include <limits.h>
29#include <errno.h>
30#include <unistd.h>
31#include <sys/stat.h>
32#include <sys/time.h>
33#include <stdio.h>
34#ifdef __DARWIN__
35# include <mach-o/dyld.h>
36#endif
37
38#include <iprt/path.h>
39#include <iprt/assert.h>
40#include <iprt/string.h>
41#include <iprt/err.h>
42#include <iprt/log.h>
43#include "internal/path.h"
44#include "internal/fs.h"
45
46#ifdef __L4__
47# include <l4/vboxserver/vboxserver.h>
48#endif
49
50
51
52
53RTDECL(int) RTPathReal(const char *pszPath, char *pszRealPath, unsigned cchRealPath)
54{
55 /*
56 * Convert input.
57 */
58 char *pszNativePath;
59 int rc = rtPathToNative(&pszNativePath, pszPath);
60 if (RT_SUCCESS(rc))
61 {
62 /*
63 * On POSIX platforms the API doesn't take a length parameter, which makes it
64 * a little bit more work.
65 */
66 char szTmpPath[PATH_MAX + 1];
67 const char *psz = realpath(pszNativePath, szTmpPath);
68 if (psz)
69 {
70 /*
71 * Convert result and copy it to the return buffer.
72 */
73 char *pszUtf8RealPath;
74 rc = rtPathFromNative(&pszUtf8RealPath, szTmpPath);
75 if (RT_SUCCESS(rc))
76 {
77 size_t cch = strlen(pszUtf8RealPath) + 1;
78 if (cch <= cchRealPath)
79 memcpy(pszRealPath, pszUtf8RealPath, cch);
80 else
81 rc = VERR_BUFFER_OVERFLOW;
82 RTStrFree(pszUtf8RealPath);
83 }
84 }
85 else
86 rc = RTErrConvertFromErrno(errno);
87 RTStrFree(pszNativePath);
88 }
89
90 LogFlow(("RTPathReal(%p:{%s}, %p:{%s}, %u): returns %Rrc\n", pszPath, pszPath,
91 pszRealPath, RT_SUCCESS(rc) ? pszRealPath : "<failed>", cchRealPath));
92 return rc;
93}
94
95
96/**
97 * Cleans up a path specifier a little bit.
98 * This includes removing duplicate slashes, uncessary single dots, and
99 * trailing slashes.
100 *
101 * @returns Number of bytes in the clean path.
102 * @param pszPath The path to cleanup.
103 * @remark Borrowed from InnoTek libc.
104 */
105static int fsCleanPath(char *pszPath)
106{
107 /*
108 * Change to '/' and remove duplicates.
109 */
110 char *pszSrc = pszPath;
111 char *pszTrg = pszPath;
112#ifdef HAVE_UNC
113 int fUnc = 0;
114 if ( RTPATH_IS_SLASH(pszPath[0])
115 && RTPATH_IS_SLASH(pszPath[1]))
116 { /* Skip first slash in a unc path. */
117 pszSrc++;
118 *pszTrg++ = '/';
119 fUnc = 1;
120 }
121#endif
122
123 for (;;)
124 {
125 char ch = *pszSrc++;
126 if (RTPATH_IS_SEP(ch))
127 {
128 *pszTrg++ = RTPATH_SLASH;
129 for (;;)
130 {
131 do ch = *pszSrc++;
132 while (RTPATH_IS_SEP(ch));
133
134 /* Remove '/./' and '/.'. */
135 if (ch != '.' || (*pszSrc && !RTPATH_IS_SEP(*pszSrc)))
136 break;
137 }
138 }
139 *pszTrg = ch;
140 if (!ch)
141 break;
142 pszTrg++;
143 }
144
145 /*
146 * Remove trailing slash if the path may be pointing to a directory.
147 */
148 int cch = pszTrg - pszPath;
149 if ( cch > 1
150 && pszTrg[-1] == RTPATH_SLASH
151#ifdef HAVE_DRIVE
152 && pszTrg[-2] != ':'
153#endif
154 && pszTrg[-2] != RTPATH_SLASH)
155 pszPath[--cch] = '\0';
156
157 return cch;
158}
159
160
161RTDECL(int) RTPathAbs(const char *pszPath, char *pszAbsPath, unsigned cchAbsPath)
162{
163 /*
164 * Convert input.
165 */
166 char *pszNativePath;
167 int rc = rtPathToNative(&pszNativePath, pszPath);
168 if (RT_FAILURE(rc))
169 {
170 LogFlow(("RTPathAbs(%p:{%s}, %p, %d): returns %Rrc\n", pszPath, pszPath, pszAbsPath, cchAbsPath));
171 return rc;
172 }
173
174 /*
175 * On POSIX platforms the API doesn't take a length parameter, which makes it
176 * a little bit more work.
177 */
178 char szTmpPath[PATH_MAX + 1];
179 char *psz = realpath(pszNativePath, szTmpPath);
180 if (!psz)
181 {
182 if (errno == ENOENT || errno == ENOTDIR)
183 {
184 if (strlen(pszNativePath) <= PATH_MAX)
185 {
186 /*
187 * Iterate the path bit by bit an apply realpath to it.
188 */
189
190 char szTmpSrc[PATH_MAX + 1];
191 strcpy(szTmpSrc, pszNativePath);
192 fsCleanPath(szTmpSrc);
193
194 size_t cch = 0; // current resolved path length
195 char *pszCur = szTmpSrc;
196
197 if (*pszCur == RTPATH_SLASH)
198 {
199 psz = szTmpPath;
200 pszCur++;
201 }
202 else
203 {
204 /* get the cwd */
205 psz = getcwd(szTmpPath, sizeof(szTmpPath));
206 AssertMsg(psz, ("Couldn't get cwd!\n"));
207 if (psz)
208 cch = strlen(psz);
209 else
210 rc = RTErrConvertFromErrno(errno);
211 }
212
213 if (psz)
214 {
215 bool fResolveSymlinks = true;
216 char szTmpPath2[PATH_MAX + 1];
217
218 while (*pszCur)
219 {
220 char *pszSlash = strchr(pszCur, RTPATH_SLASH);
221 size_t cchElement = pszSlash ? pszSlash - pszCur : strlen(pszCur);
222 if (cch + cchElement + 1 > PATH_MAX)
223 {
224 rc = VERR_FILENAME_TOO_LONG;
225 break;
226 }
227
228 if (!strncmp(pszCur, "..", cchElement))
229 {
230 char *pszLastSlash = strrchr(psz, RTPATH_SLASH);
231 if (pszLastSlash)
232 {
233 cch = pszLastSlash - psz;
234 psz[cch] = '\0';
235 }
236 /* else: We've reached the root and the parent of the root is the root. */
237 }
238 else
239 {
240 psz[cch++] = RTPATH_SLASH;
241 memcpy(psz + cch, pszCur, cchElement);
242 cch += cchElement;
243 psz[cch] = '\0';
244
245 if (fResolveSymlinks)
246 {
247 /* resolve possible symlinks */
248 char *psz2 = realpath(psz, psz == szTmpPath
249 ? szTmpPath2
250 : szTmpPath);
251 if (psz2)
252 {
253 psz = psz2;
254 cch = strlen(psz);
255 }
256 else
257 {
258 if (errno != ENOENT && errno != ENOTDIR)
259 {
260 rc = RTErrConvertFromErrno(errno);
261 break;
262 }
263
264 /* no more need to resolve symlinks */
265 fResolveSymlinks = false;
266 }
267 }
268 }
269
270 pszCur += cchElement;
271 /* skip the slash */
272 if (*pszCur)
273 ++pszCur;
274 }
275
276 /* if the length is zero here, then we're at the root (Not true for half-posixs stuff such as libc!) */
277 if (!cch)
278 {
279 psz[cch++] = RTPATH_SLASH;
280 psz[cch] = '\0';
281 }
282 }
283 }
284 else
285 rc = VERR_FILENAME_TOO_LONG;
286 }
287 else
288 rc = RTErrConvertFromErrno(errno);
289 }
290
291 RTStrFree(pszNativePath);
292
293 if (psz && RT_SUCCESS(rc))
294 {
295 /*
296 * Convert result and copy it to the return buffer.
297 */
298 char *pszUtf8AbsPath;
299 rc = rtPathFromNative(&pszUtf8AbsPath, psz);
300 if (RT_FAILURE(rc))
301 {
302 LogFlow(("RTPathAbs(%p:{%s}, %p, %d): returns %Rrc\n", pszPath, pszPath, pszAbsPath, cchAbsPath));
303 return rc;
304 }
305
306 unsigned cch = strlen(pszUtf8AbsPath) + 1;
307 if (cch <= cchAbsPath)
308 memcpy(pszAbsPath, pszUtf8AbsPath, cch);
309 else
310 rc = VERR_BUFFER_OVERFLOW;
311 RTStrFree(pszUtf8AbsPath);
312 }
313
314 LogFlow(("RTPathAbs(%p:{%s}, %p:{%s}, %d): returns %Rrc\n", pszPath, pszPath,
315 pszAbsPath, RT_SUCCESS(rc) ? pszAbsPath : "<failed>", cchAbsPath));
316 return rc;
317}
318
319
320RTDECL(int) RTPathProgram(char *pszPath, unsigned cchPath)
321{
322 /*
323 * First time only.
324 */
325 if (!g_szrtProgramPath[0])
326 {
327 /*
328 * Linux have no API for obtaining the executable path, but provides a symbolic link
329 * in the proc file system. Note that readlink is one of the weirdest Unix apis around.
330 *
331 * OS/2 have an api for getting the program file name.
332 */
333/** @todo use RTProcGetExecutableName() */
334#if defined(__LINUX__) || defined(__FREEBSD__)
335# ifdef __LINUX__
336 int cchLink = readlink("/proc/self/exe", &g_szrtProgramPath[0], sizeof(g_szrtProgramPath) - 1);
337# else
338 int cchLink = readlink("/proc/curproc/file", &g_szrtProgramPath[0], sizeof(g_szrtProgramPath) - 1);
339# endif
340 if (cchLink < 0 || cchLink == sizeof(g_szrtProgramPath) - 1)
341 {
342 int rc = RTErrConvertFromErrno(errno);
343 AssertMsgFailed(("couldn't read /proc/self/exe. errno=%d cchLink=%d\n", errno, cchLink));
344 LogFlow(("RTPathProgram(%p, %u): returns %Rrc\n", pszPath, cchPath, rc));
345 return rc;
346 }
347 g_szrtProgramPath[cchLink] = '\0';
348
349#elif defined(__OS2__) || defined(__L4__)
350 _execname(g_szrtProgramPath, sizeof(g_szrtProgramPath));
351
352#elif defined(__DARWIN__)
353 const char *pszImageName = _dyld_get_image_name(0);
354 AssertReturn(pszImageName, VERR_INTERNAL_ERROR);
355 size_t cchImageName = strlen(pszImageName);
356 if (cchImageName >= sizeof(g_szrtProgramPath))
357 AssertReturn(pszImageName, VERR_INTERNAL_ERROR);
358 memcpy(g_szrtProgramPath, pszImageName, cchImageName + 1);
359
360#else
361# error needs porting.
362#endif
363
364 /*
365 * Convert to UTF-8 and strip of the filename.
366 */
367 char *pszTmp = NULL;
368 int rc = rtPathFromNative(&pszTmp, &g_szrtProgramPath[0]);
369 if (RT_FAILURE(rc))
370 {
371 LogFlow(("RTPathProgram(%p, %u): returns %Rrc\n", pszPath, cchPath, rc));
372 return rc;
373 }
374 size_t cch = strlen(pszTmp);
375 if (cch >= sizeof(g_szrtProgramPath))
376 {
377 RTStrFree(pszTmp);
378 LogFlow(("RTPathProgram(%p, %u): returns %Rrc\n", pszPath, cchPath, VERR_BUFFER_OVERFLOW));
379 return VERR_BUFFER_OVERFLOW;
380 }
381 memcpy(g_szrtProgramPath, pszTmp, cch + 1);
382 RTPathStripFilename(g_szrtProgramPath);
383 RTStrFree(pszTmp);
384 }
385
386 /*
387 * Calc the length and check if there is space before copying.
388 */
389 unsigned cch = strlen(g_szrtProgramPath) + 1;
390 if (cch <= cchPath)
391 {
392 memcpy(pszPath, g_szrtProgramPath, cch + 1);
393 LogFlow(("RTPathProgram(%p:{%s}, %u): returns %Rrc\n", pszPath, pszPath, cchPath, VINF_SUCCESS));
394 return VINF_SUCCESS;
395 }
396
397 AssertMsgFailed(("Buffer too small (%d < %d)\n", cchPath, cch));
398 LogFlow(("RTPathProgram(%p, %u): returns %Rrc\n", pszPath, cchPath, VERR_BUFFER_OVERFLOW));
399 return VERR_BUFFER_OVERFLOW;
400}
401
402
403RTDECL(int) RTPathUserHome(char *pszPath, unsigned cchPath)
404{
405 /*
406 * Get HOME env. var it and validate it's existance.
407 */
408 int rc;
409 struct stat s;
410 const char *pszHome = getenv("HOME");
411 if (pszHome)
412 {
413 if ( !stat(pszHome, &s)
414 && S_ISDIR(s.st_mode))
415 {
416 /*
417 * Convert it to UTF-8 and copy it to the return buffer.
418 */
419 char *pszUtf8Path;
420 rc = rtPathFromNative(&pszUtf8Path, pszHome);
421 if (RT_SUCCESS(rc))
422 {
423 size_t cchHome = strlen(pszUtf8Path);
424 if (cchHome < cchPath)
425 memcpy(pszPath, pszUtf8Path, cchHome + 1);
426 else
427 rc = VERR_BUFFER_OVERFLOW;
428 RTStrFree(pszUtf8Path);
429 }
430 }
431 else
432 rc = VERR_PATH_NOT_FOUND;
433
434 }
435 else
436 rc = VERR_PATH_NOT_FOUND;
437
438 LogFlow(("RTPathUserHome(%p:{%s}, %u): returns %Rrc\n", pszPath,
439 RT_SUCCESS(rc) ? pszPath : "<failed>", cchPath, rc));
440 return rc;
441}
442
443
444RTR3DECL(int) RTPathQueryInfo(const char *pszPath, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAdditionalAttribs)
445{
446 /*
447 * Validate input.
448 */
449 AssertMsgReturn(VALID_PTR(pszPath), ("%p\n", pszPath), VERR_INVALID_POINTER);
450 AssertReturn(*pszPath, VERR_INVALID_PARAMETER);
451 AssertMsgReturn(VALID_PTR(pObjInfo), ("%p\n", pszPath), VERR_INVALID_POINTER);
452 AssertMsgReturn( enmAdditionalAttribs >= RTFSOBJATTRADD_NOTHING
453 && enmAdditionalAttribs <= RTFSOBJATTRADD_LAST,
454 ("Invalid enmAdditionalAttribs=%p\n", enmAdditionalAttribs),
455 VERR_INVALID_PARAMETER);
456
457 /*
458 * Convert the filename.
459 */
460 char *pszNativePath;
461 int rc = rtPathToNative(&pszNativePath, pszPath);
462 if (RT_SUCCESS(rc))
463 {
464 struct stat Stat;
465 if (!stat(pszNativePath, &Stat))
466 {
467 rtFsConvertStatToObjInfo(pObjInfo, &Stat);
468 switch (enmAdditionalAttribs)
469 {
470 case RTFSOBJATTRADD_EASIZE:
471 /** @todo Use SGI extended attribute interface to query EA info. */
472 pObjInfo->Attr.enmAdditional = RTFSOBJATTRADD_EASIZE;
473 pObjInfo->Attr.u.EASize.cb = 0;
474 break;
475
476 case RTFSOBJATTRADD_NOTHING:
477 case RTFSOBJATTRADD_UNIX:
478 Assert(pObjInfo->Attr.enmAdditional == RTFSOBJATTRADD_UNIX);
479 break;
480
481 default:
482 AssertMsgFailed(("Impossible!\n"));
483 return VERR_INTERNAL_ERROR;
484 }
485 }
486 else
487 rc = RTErrConvertFromErrno(errno);
488 }
489
490 LogFlow(("RTPathQueryInfo(%p:{%s}, pObjInfo=%p, %d): returns %Rrc\n",
491 pszPath, pszPath, pObjInfo, enmAdditionalAttribs, rc));
492 return rc;
493}
494
495
496RTR3DECL(int) RTPathSetTimes(const char *pszPath, PCRTTIMESPEC pAccessTime, PCRTTIMESPEC pModificationTime,
497 PCRTTIMESPEC pChangeTime, PCRTTIMESPEC pBirthTime)
498{
499 /*
500 * Validate input.
501 */
502 AssertMsgReturn(VALID_PTR(pszPath), ("%p\n", pszPath), VERR_INVALID_POINTER);
503 AssertMsgReturn(*pszPath, ("%p\n", pszPath), VERR_INVALID_PARAMETER);
504 AssertMsgReturn(!pAccessTime || VALID_PTR(pAccessTime), ("%p\n", pAccessTime), VERR_INVALID_POINTER);
505 AssertMsgReturn(!pModificationTime || VALID_PTR(pModificationTime), ("%p\n", pModificationTime), VERR_INVALID_POINTER);
506 AssertMsgReturn(!pChangeTime || VALID_PTR(pChangeTime), ("%p\n", pChangeTime), VERR_INVALID_POINTER);
507 AssertMsgReturn(!pBirthTime || VALID_PTR(pBirthTime), ("%p\n", pBirthTime), VERR_INVALID_POINTER);
508
509 /*
510 * Convert the paths.
511 */
512 char *pszNativePath;
513 int rc = rtPathToNative(&pszNativePath, pszPath);
514 if (RT_SUCCESS(rc))
515 {
516 /*
517 * If it's a no-op, we'll only verify the existance of the file.
518 */
519 if (!pAccessTime && !pModificationTime)
520 {
521 struct stat Stat;
522 if (!stat(pszNativePath, &Stat))
523 rc = VINF_SUCCESS;
524 else
525 {
526 rc = RTErrConvertFromErrno(errno);
527 Log(("RTPathSetTimes('%s',,,,): failed with %Rrc and errno=%d\n", pszPath, rc, errno));
528 }
529 }
530 else
531 {
532 /*
533 * Convert the input to timeval, getting the missing one if necessary,
534 * and call the API which does the change.
535 */
536 struct timeval aTimevals[2];
537 if (pAccessTime && pModificationTime)
538 {
539 RTTimeSpecGetTimeval(pAccessTime, &aTimevals[0]);
540 RTTimeSpecGetTimeval(pModificationTime, &aTimevals[1]);
541 }
542 else
543 {
544 RTFSOBJINFO ObjInfo;
545 int rc = RTPathQueryInfo(pszPath, &ObjInfo, RTFSOBJATTRADD_UNIX);
546 if (RT_SUCCESS(rc))
547 {
548 RTTimeSpecGetTimeval(pAccessTime ? pAccessTime : &ObjInfo.AccessTime, &aTimevals[0]);
549 RTTimeSpecGetTimeval(pModificationTime ? pModificationTime : &ObjInfo.ModificationTime, &aTimevals[1]);
550 }
551 else
552 Log(("RTPathSetTimes('%s',%p,%p,,): RTPathQueryInfo failed with %Rrc\n",
553 pszPath, pAccessTime, pModificationTime, rc));
554 }
555 if (RT_SUCCESS(rc))
556 {
557 if (utimes(pszNativePath, aTimevals))
558 {
559 rc = RTErrConvertFromErrno(errno);
560 Log(("RTPathSetTimes('%s',%p,%p,,): failed with %Rrc and errno=%d\n",
561 pszPath, pAccessTime, pModificationTime, rc, errno));
562 }
563 }
564 }
565 }
566
567 LogFlow(("RTPathSetTimes(%p:{%s}, %p:{%RDtimespec}, %p:{%RDtimespec}, %p:{%RDtimespec}, %p:{%RDtimespec}): return %Rrc\n",
568 pszPath, pszPath, pAccessTime, pAccessTime, pModificationTime, pModificationTime,
569 pChangeTime, pChangeTime, pBirthTime, pBirthTime));
570 return rc;
571}
572
573
574/**
575 * Checks if two files are the one and same file.
576 */
577static bool rtPathSame(const char *pszNativeSrc, const char *pszNativeDst)
578{
579 struct stat SrcStat;
580 if (stat(pszNativeSrc, &SrcStat))
581 return false;
582 struct stat DstStat;
583 if (stat(pszNativeDst, &DstStat))
584 return false;
585 Assert(SrcStat.st_dev && DstStat.st_dev);
586 Assert(SrcStat.st_ino && DstStat.st_ino);
587 if ( SrcStat.st_dev == DstStat.st_dev
588 && SrcStat.st_ino == DstStat.st_ino
589 && (SrcStat.st_mode & S_IFMT) == (DstStat.st_mode & S_IFMT))
590 return true;
591 return false;
592}
593
594
595/**
596 * Worker for RTPathRename, RTDirRename, RTFileRename.
597 *
598 * @returns IPRT status code.
599 * @param pszSrc The source path.
600 * @param pszDst The destintation path.
601 * @param fRename The rename flags.
602 * @param fFileType The filetype. We use the RTFMODE filetypes here. If it's 0,
603 * anything goes. If it's RTFS_TYPE_DIRECTORY we'll check that the
604 * source is a directory. If Its RTFS_TYPE_FILE we'll check that it's
605 * not a directory (we are NOT checking whether it's a file).
606 */
607int rtPathPosixRename(const char *pszSrc, const char *pszDst, unsigned fRename, RTFMODE fFileType)
608{
609 /*
610 * Convert the paths.
611 */
612 char *pszNativeSrc;
613 int rc = rtPathToNative(&pszNativeSrc, pszSrc);
614 if (RT_SUCCESS(rc))
615 {
616 char *pszNativeDst;
617 rc = rtPathToNative(&pszNativeDst, pszDst);
618 if (RT_SUCCESS(rc))
619 {
620 /*
621 * Check that the source exists and that any types that's specified matches.
622 * We have to check this first to avoid getting errnous VERR_ALREADY_EXISTS
623 * errors from the next step.
624 *
625 * There are race conditions here (perhaps unlikly ones but still), but I'm
626 * afraid there is little with can do to fix that.
627 */
628 struct stat SrcStat;
629 if (stat(pszNativeSrc, &SrcStat))
630 rc = RTErrConvertFromErrno(errno);
631 else if (!fFileType)
632 rc = VINF_SUCCESS;
633 else if (RTFS_IS_DIRECTORY(fFileType))
634 rc = S_ISDIR(SrcStat.st_mode) ? VINF_SUCCESS : VERR_NOT_A_DIRECTORY;
635 else
636 rc = S_ISDIR(SrcStat.st_mode) ? VERR_IS_A_DIRECTORY : VINF_SUCCESS;
637 if (RT_SUCCESS(rc))
638 {
639 bool fSameFile = false;
640
641 /*
642 * Check if the target exists, rename is rather destructive.
643 * We'll have to make sure we don't overwrite the source!
644 * Another race condition btw.
645 */
646 struct stat DstStat;
647 if (stat(pszNativeDst, &DstStat))
648 rc = errno == ENOENT ? VINF_SUCCESS : RTErrConvertFromErrno(errno);
649 else
650 {
651 Assert(SrcStat.st_dev && DstStat.st_dev);
652 Assert(SrcStat.st_ino && DstStat.st_ino);
653 if ( SrcStat.st_dev == DstStat.st_dev
654 && SrcStat.st_ino == DstStat.st_ino
655 && (SrcStat.st_mode & S_IFMT) == (SrcStat.st_mode & S_IFMT))
656 {
657 /*
658 * It's likely that we're talking about the same file here.
659 * We should probably check paths or whatever, but for now this'll have to be enough.
660 */
661 fSameFile = true;
662 }
663 if (fSameFile)
664 rc = VINF_SUCCESS;
665 else if (S_ISDIR(DstStat.st_mode) || !(fRename & RTPATHRENAME_FLAGS_REPLACE))
666 rc = VERR_ALREADY_EXISTS;
667 else
668 rc = VINF_SUCCESS;
669
670 }
671 if (RT_SUCCESS(rc))
672 {
673 if (!rename(pszNativeSrc, pszNativeDst))
674 rc = VINF_SUCCESS;
675 else if ( (fRename & RTPATHRENAME_FLAGS_REPLACE)
676 && (errno == ENOTDIR || errno == EEXIST))
677 {
678 /*
679 * Check that the destination isn't a directory.
680 * Yet another race condition.
681 */
682 if (rtPathSame(pszNativeSrc, pszNativeDst))
683 {
684 rc = VINF_SUCCESS;
685 Log(("rtPathRename('%s', '%s', %#x ,%RTfmode): appears to be the same file... (errno=%d)\n",
686 pszSrc, pszDst, fRename, fFileType, errno));
687 }
688 else
689 {
690 if (stat(pszNativeDst, &DstStat))
691 rc = errno != ENOENT ? RTErrConvertFromErrno(errno) : VINF_SUCCESS;
692 else if (S_ISDIR(DstStat.st_mode))
693 rc = VERR_ALREADY_EXISTS;
694 else
695 rc = VINF_SUCCESS;
696 if (RT_SUCCESS(rc))
697 {
698 if (!unlink(pszNativeDst))
699 {
700 if (!rename(pszNativeSrc, pszNativeDst))
701 rc = VINF_SUCCESS;
702 else
703 {
704 rc = RTErrConvertFromErrno(errno);
705 Log(("rtPathRename('%s', '%s', %#x ,%RTfmode): rename failed rc=%Rrc errno=%d\n",
706 pszSrc, pszDst, fRename, fFileType, rc, errno));
707 }
708 }
709 else
710 {
711 rc = RTErrConvertFromErrno(errno);
712 Log(("rtPathRename('%s', '%s', %#x ,%RTfmode): failed to unlink dst rc=%Rrc errno=%d\n",
713 pszSrc, pszDst, fRename, fFileType, rc, errno));
714 }
715 }
716 else
717 Log(("rtPathRename('%s', '%s', %#x ,%RTfmode): dst !dir check failed rc=%Rrc\n",
718 pszSrc, pszDst, fRename, fFileType, rc));
719 }
720 }
721 else
722 {
723 rc = RTErrConvertFromErrno(errno);
724 if (errno == ENOTDIR)
725 rc = VERR_ALREADY_EXISTS; /* unless somebody is racing us, this is the right interpretation */
726 Log(("rtPathRename('%s', '%s', %#x ,%RTfmode): rename failed rc=%Rrc errno=%d\n",
727 pszSrc, pszDst, fRename, fFileType, rc, errno));
728 }
729 }
730 else
731 Log(("rtPathRename('%s', '%s', %#x ,%RTfmode): destination check failed rc=%Rrc errno=%d\n",
732 pszSrc, pszDst, fRename, fFileType, rc, errno));
733 }
734 else
735 Log(("rtPathRename('%s', '%s', %#x ,%RTfmode): source type check failed rc=%Rrc errno=%d\n",
736 pszSrc, pszDst, fRename, fFileType, rc, errno));
737
738 rtPathFreeNative(pszNativeDst);
739 }
740 rtPathFreeNative(pszNativeSrc);
741 }
742 return rc;
743}
744
745
746RTR3DECL(int) RTPathRename(const char *pszSrc, const char *pszDst, unsigned fRename)
747{
748 /*
749 * Validate input.
750 */
751 AssertMsgReturn(VALID_PTR(pszSrc), ("%p\n", pszSrc), VERR_INVALID_POINTER);
752 AssertMsgReturn(VALID_PTR(pszDst), ("%p\n", pszDst), VERR_INVALID_POINTER);
753 AssertMsgReturn(*pszSrc, ("%p\n", pszSrc), VERR_INVALID_PARAMETER);
754 AssertMsgReturn(*pszDst, ("%p\n", pszDst), VERR_INVALID_PARAMETER);
755 AssertMsgReturn(!(fRename & ~RTPATHRENAME_FLAGS_REPLACE), ("%#x\n", fRename), VERR_INVALID_PARAMETER);
756
757 /*
758 * Hand it to the worker.
759 */
760 int rc = rtPathPosixRename(pszSrc, pszDst, fRename, 0);
761
762 Log(("RTPathRename(%p:{%s}, %p:{%s}, %#x): returns %Rrc\n", pszSrc, pszSrc, pszDst, pszDst, fRename, rc));
763 return rc;
764}
765
766
767RTDECL(bool) RTPathExists(const char *pszPath)
768{
769 /*
770 * Validate input.
771 */
772 AssertPtrReturn(pszPath, false);
773 AssertReturn(*pszPath, false);
774
775 /*
776 * Convert the path and check if it exists using stat().
777 */
778 char *pszNativePath;
779 int rc = rtPathToNative(&pszNativePath, pszPath);
780 if (RT_SUCCESS(rc))
781 {
782 struct stat Stat;
783 if (!stat(pszNativePath, &Stat))
784 rc = VINF_SUCCESS;
785 else
786 rc = VERR_GENERAL_FAILURE;
787 RTStrFree(pszNativePath);
788 }
789 return RT_SUCCESS(rc);
790}
791
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