VirtualBox

source: vbox/trunk/src/VBox/Runtime/testcase/tstRTPath.cpp@ 93411

Last change on this file since 93411 was 93115, checked in by vboxsync, 3 years ago

scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 61.7 KB
Line 
1/* $Id: tstRTPath.cpp 93115 2022-01-01 11:31:46Z vboxsync $ */
2/** @file
3 * IPRT Testcase - Test various path functions.
4 */
5
6/*
7 * Copyright (C) 2006-2022 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#include <iprt/path.h>
32
33#include <iprt/err.h>
34#include <iprt/initterm.h>
35#include <iprt/param.h>
36#include <iprt/process.h>
37#include <iprt/stream.h>
38#include <iprt/string.h>
39#include <iprt/test.h>
40
41
42static void testParserAndSplitter(RTTEST hTest)
43{
44 static struct
45 {
46 uint16_t cComps;
47 uint16_t cchPath;
48 uint16_t offSuffix;
49 int16_t offName; /**< RTPathParseSimple */
50 uint16_t cchDir; /**< RTPathParseSimple */
51 const char *pszPath;
52 uint16_t fProps;
53 uint32_t fFlags;
54 } const s_aTests[] =
55 {
56 { 2, 5, 5, -1, 4, "/bin/", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_DIR_SLASH, RTPATH_STR_F_STYLE_UNIX },
57 { 2, 13, 9, 3, 3, "C:/Config.sys", RTPATH_PROP_VOLUME | RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME | RTPATH_PROP_SUFFIX, RTPATH_STR_F_STYLE_DOS },
58 { 2, 13, 10, 4, 4, "C://Config.sys", RTPATH_PROP_VOLUME | RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME | RTPATH_PROP_SUFFIX | RTPATH_PROP_EXTRA_SLASHES, RTPATH_STR_F_STYLE_DOS },
59 { 2, 12, 8, 2, 2, "C:Config.sys", RTPATH_PROP_VOLUME | RTPATH_PROP_RELATIVE | RTPATH_PROP_FILENAME | RTPATH_PROP_SUFFIX, RTPATH_STR_F_STYLE_DOS },
60 { 1, 10, 6, 0, 0, "Config.sys", RTPATH_PROP_RELATIVE | RTPATH_PROP_FILENAME | RTPATH_PROP_SUFFIX, RTPATH_STR_F_STYLE_DOS },
61 { 3, 15, 11, 7, 6, "C:/Win/file.ext", RTPATH_PROP_VOLUME | RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME | RTPATH_PROP_SUFFIX , RTPATH_STR_F_STYLE_DOS },
62 { 1, 4, 4, -1, 4, "//./", RTPATH_PROP_UNC | RTPATH_PROP_SPECIAL_UNC | RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE, RTPATH_STR_F_STYLE_DOS },
63 { 2, 5, 5, 4, 4, "//./f", RTPATH_PROP_UNC | RTPATH_PROP_SPECIAL_UNC | RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME, RTPATH_STR_F_STYLE_DOS },
64 { 2, 5, 6, 5, 5, "//.//f", RTPATH_PROP_UNC | RTPATH_PROP_SPECIAL_UNC | RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME | RTPATH_PROP_EXTRA_SLASHES, RTPATH_STR_F_STYLE_DOS },
65 { 3, 7, 7, 6, 5, "//././f", RTPATH_PROP_UNC | RTPATH_PROP_SPECIAL_UNC | RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME | RTPATH_PROP_DOT_REFS, RTPATH_STR_F_STYLE_DOS },
66 { 3, 8, 8, 7, 6, "//.././f", RTPATH_PROP_UNC | RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME | RTPATH_PROP_DOT_REFS, RTPATH_STR_F_STYLE_DOS },
67 { 3, 9, 9, 8, 7, "//../../f", RTPATH_PROP_UNC | RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_RELATIVE | RTPATH_PROP_FILENAME | RTPATH_PROP_DOTDOT_REFS, RTPATH_STR_F_STYLE_DOS },
68 { 1, 1, 1, -1, 1, "/", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE, RTPATH_STR_F_STYLE_UNIX },
69 { 2, 4, 4, 1, 1, "/bin", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME, RTPATH_STR_F_STYLE_UNIX },
70 { 2, 5, 5, -1, 4, "/bin/", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_DIR_SLASH, RTPATH_STR_F_STYLE_UNIX },
71 { 3, 7, 7, 5, 4, "/bin/ls", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME, RTPATH_STR_F_STYLE_UNIX },
72 { 3, 12, 7, 5, 4, "/etc/rc.conf", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME | RTPATH_PROP_SUFFIX, RTPATH_STR_F_STYLE_UNIX },
73 { 1, 1, 2, -1, 2, "//", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_EXTRA_SLASHES, RTPATH_STR_F_STYLE_UNIX },
74 { 1, 1, 3, -1, 3, "///", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_EXTRA_SLASHES, RTPATH_STR_F_STYLE_UNIX },
75 { 3, 6, 7, 4, 2, "/.//bin", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_EXTRA_SLASHES | RTPATH_PROP_DOT_REFS | RTPATH_PROP_FILENAME, RTPATH_STR_F_STYLE_UNIX },
76 { 1, 3, 3, 0, 0, "bin", RTPATH_PROP_RELATIVE | RTPATH_PROP_FILENAME, RTPATH_STR_F_STYLE_UNIX },
77 { 1, 4, 4, -1, 3, "bin/", RTPATH_PROP_RELATIVE | RTPATH_PROP_DIR_SLASH, RTPATH_STR_F_STYLE_UNIX },
78 { 1, 4, 7, -1, 3, "bin////", RTPATH_PROP_RELATIVE | RTPATH_PROP_DIR_SLASH | RTPATH_PROP_EXTRA_SLASHES, RTPATH_STR_F_STYLE_UNIX },
79 { 3, 10, 10, 7, 6, "bin/../usr", RTPATH_PROP_RELATIVE | RTPATH_PROP_DOTDOT_REFS | RTPATH_PROP_FILENAME, RTPATH_STR_F_STYLE_UNIX },
80 { 4, 11, 11, 8, 7, "/bin/../usr", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_RELATIVE | RTPATH_PROP_DOTDOT_REFS | RTPATH_PROP_FILENAME, RTPATH_STR_F_STYLE_UNIX },
81 { 4, 8, 8, 7, 6, "/a/.../u", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME, RTPATH_STR_F_STYLE_UNIX },
82 { 4, 8, 8, 7, 6, "/a/.b./u", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME, RTPATH_STR_F_STYLE_UNIX },
83 { 4, 8, 8, 7, 6, "/a/..c/u", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME, RTPATH_STR_F_STYLE_UNIX },
84 { 4, 8, 8, 7, 6, "/a/d../u", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME, RTPATH_STR_F_STYLE_UNIX },
85 { 4, 8, 8, 6, 5, "/a/.e/.u", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME, RTPATH_STR_F_STYLE_UNIX },
86 { 4, 8, 8, 6, 5, "/a/.f/.u", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME, RTPATH_STR_F_STYLE_UNIX },
87 { 4, 11, 7, 6, 5, "/a/.f/u.ext", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME | RTPATH_PROP_SUFFIX, RTPATH_STR_F_STYLE_UNIX },
88 { 4, 8, 8, 6, 5, "/a/.g/u.", RTPATH_PROP_ROOT_SLASH | RTPATH_PROP_ABSOLUTE | RTPATH_PROP_FILENAME, RTPATH_STR_F_STYLE_UNIX },
89 { 3, 9, 10, 5, 4, "/a/h/u.ext", RTPATH_PROP_EXTRA_SLASHES | RTPATH_PROP_RELATIVE, RTPATH_STR_F_STYLE_UNIX | RTPATH_STR_F_MIDDLE },
90 { 3, 9, 9, 5, 3, "a/h/u.ext", RTPATH_PROP_RELATIVE, RTPATH_STR_F_STYLE_UNIX | RTPATH_STR_F_MIDDLE },
91 { 3, 9, 10, -1, 10, "a/h/u.ext/", RTPATH_PROP_EXTRA_SLASHES | RTPATH_PROP_RELATIVE, RTPATH_STR_F_STYLE_UNIX | RTPATH_STR_F_MIDDLE },
92 };
93
94 char szPath1[RTPATH_MAX];
95 union
96 {
97 RTPATHPARSED Parsed;
98 RTPATHSPLIT Split;
99 uint8_t ab[4096];
100 } u;
101
102 RTTestSub(hTest, "RTPathParse");
103 for (uint32_t i = 0; i < RT_ELEMENTS(s_aTests); i++)
104 {
105 memset(&u, i & 1 ? 0xff : 0, sizeof(u));
106 int rc = RTPathParse(s_aTests[i].pszPath, &u.Parsed, sizeof(u), s_aTests[i].fFlags);
107 if ( rc != VINF_SUCCESS
108 || s_aTests[i].cComps != u.Parsed.cComps
109 || s_aTests[i].fProps != u.Parsed.fProps
110 || s_aTests[i].offSuffix != u.Parsed.offSuffix
111 || s_aTests[i].cchPath != u.Parsed.cchPath)
112 {
113 RTTestFailed(hTest, "i=%d rc=%Rrc %s", i, rc, s_aTests[i].pszPath);
114 RTTestFailureDetails(hTest,
115 " cComps %u, got %u\n"
116 " fProps %#x, got %#x, xor=>%#x\n"
117 " offSuffix %u, got %u\n"
118 " cchPath %u, got %u\n"
119 ,
120 s_aTests[i].cComps, u.Parsed.cComps,
121 s_aTests[i].fProps, u.Parsed.fProps, s_aTests[i].fProps ^ u.Parsed.fProps,
122 s_aTests[i].offSuffix, u.Parsed.offSuffix,
123 s_aTests[i].cchPath, u.Parsed.cchPath);
124 }
125 else
126 {
127 rc = RTPathParsedReassemble(s_aTests[i].pszPath, &u.Parsed, s_aTests[i].fFlags & ~RTPATH_STR_F_MIDDLE,
128 szPath1, sizeof(szPath1));
129 if (rc == VINF_SUCCESS)
130 {
131 RTTESTI_CHECK_MSG(strlen(szPath1) == s_aTests[i].cchPath, ("%s\n", szPath1));
132 if ( !(u.Parsed.fProps & RTPATH_PROP_EXTRA_SLASHES)
133 && (s_aTests[i].fFlags & RTPATH_STR_F_STYLE_MASK) != RTPATH_STR_F_STYLE_DOS)
134 RTTESTI_CHECK_MSG(strcmp(szPath1, s_aTests[i].pszPath) == 0, ("%s\n", szPath1));
135 }
136 else
137 RTTestIFailed("RTPathParsedReassemble -> %Rrc", rc);
138 }
139 }
140
141 RTTestSub(hTest, "RTPathSplit");
142 for (uint32_t i = 0; i < RT_ELEMENTS(s_aTests); i++)
143 {
144 memset(&u, i & 1 ? 0xff : 0, sizeof(u));
145 int rc = RTPathSplit(s_aTests[i].pszPath, &u.Split, sizeof(u), s_aTests[i].fFlags);
146 if ( rc != VINF_SUCCESS
147 || s_aTests[i].cComps != u.Split.cComps
148 || s_aTests[i].fProps != u.Split.fProps
149 || s_aTests[i].cchPath != u.Split.cchPath)
150 {
151 RTTestFailed(hTest, "i=%d rc=%Rrc %s", i, rc, s_aTests[i].pszPath);
152 RTTestFailureDetails(hTest,
153 " cComps %u, got %u\n"
154 " fProps %#x, got %#x, xor=>%#x\n"
155 " cchPath %u, got %u\n"
156 ,
157 s_aTests[i].cComps, u.Split.cComps,
158 s_aTests[i].fProps, u.Split.fProps, s_aTests[i].fProps ^ u.Split.fProps,
159 s_aTests[i].cchPath, u.Split.cchPath);
160 }
161 else
162 {
163 RTTESTI_CHECK_MSG(*u.Split.pszSuffix == '\0' || *u.Split.pszSuffix == '.', ("%s", u.Split.pszSuffix));
164 for (uint32_t idxComp = RTPATH_PROP_HAS_ROOT_SPEC(u.Split.fProps); idxComp < u.Split.cComps; idxComp++)
165 if ( (s_aTests[i].fFlags & RTPATH_STR_F_STYLE_MASK) == RTPATH_STR_F_STYLE_DOS
166 ? strpbrk(u.Split.apszComps[idxComp], "/\\")
167 : strchr(u.Split.apszComps[idxComp], RTPATH_SLASH) )
168 RTTestFailed(hTest, "i=%d idxComp=%d '%s'", i, idxComp, u.Split.apszComps[idxComp]);
169
170 PRTPATHSPLIT pSplit = NULL;
171 RTTESTI_CHECK_RC(rc = RTPathSplitA(s_aTests[i].pszPath, &pSplit, s_aTests[i].fFlags), VINF_SUCCESS);
172 if (RT_SUCCESS(rc))
173 {
174 RTTESTI_CHECK(pSplit);
175 RTTESTI_CHECK(pSplit->cComps == u.Split.cComps);
176 RTTESTI_CHECK(pSplit->fProps == u.Split.fProps);
177 RTTESTI_CHECK(pSplit->cchPath == u.Split.cchPath);
178 RTTESTI_CHECK(pSplit->cbNeeded == u.Split.cbNeeded);
179 RTTESTI_CHECK(!strcmp(pSplit->pszSuffix, u.Split.pszSuffix));
180 for (uint32_t idxComp = 0; idxComp < u.Split.cComps; idxComp++)
181 RTTESTI_CHECK(!strcmp(pSplit->apszComps[idxComp], u.Split.apszComps[idxComp]));
182 RTPathSplitFree(pSplit);
183 }
184
185 rc = RTPathSplitReassemble(&u.Split, s_aTests[i].fFlags & ~RTPATH_STR_F_MIDDLE, szPath1, sizeof(szPath1));
186 if (rc == VINF_SUCCESS)
187 {
188 RTTESTI_CHECK_MSG(strlen(szPath1) == s_aTests[i].cchPath, ("%s\n", szPath1));
189 if ( !(u.Parsed.fProps & RTPATH_PROP_EXTRA_SLASHES)
190 && (s_aTests[i].fFlags & RTPATH_STR_F_STYLE_MASK) != RTPATH_STR_F_STYLE_DOS)
191 RTTESTI_CHECK_MSG(strcmp(szPath1, s_aTests[i].pszPath) == 0, ("%s\n", szPath1));
192 }
193 else
194 RTTestIFailed("RTPathSplitReassemble -> %Rrc", rc);
195 }
196 }
197
198 RTTestSub(hTest, "RTPathParseSimple");
199 for (uint32_t i = 0; i < RT_ELEMENTS(s_aTests); i++)
200 {
201 if ( (s_aTests[i].fFlags & RTPATH_STR_F_STYLE_MASK) != RTPATH_STR_F_STYLE_HOST
202 && (s_aTests[i].fFlags & RTPATH_STR_F_STYLE_MASK) != RTPATH_STYLE)
203 continue;
204 if (s_aTests[i].fFlags & ~RTPATH_STR_F_STYLE_MASK)
205 continue;
206
207 size_t const cchPathIn = strlen(s_aTests[i].pszPath);
208 size_t cchDir = ~(size_t)1;
209 ssize_t offName = -97;
210 ssize_t offSuff = -99;
211 size_t cchPath = RTPathParseSimple(s_aTests[i].pszPath, &cchDir, &offName, &offSuff);
212 if ( cchPath != cchPathIn
213 || offSuff != (s_aTests[i].offSuffix >= cchPathIn ? -1 : s_aTests[i].offSuffix)
214 || offName != s_aTests[i].offName
215 || cchDir != s_aTests[i].cchDir)
216 {
217 RTTestFailed(hTest, "i=%u %s", i, s_aTests[i].pszPath);
218 RTTestFailureDetails(hTest,
219 " cchPath %zu, got %zu\n"
220 " cchDir %u, got %zu\n"
221 " offName %d, got %zd\n"
222 " offSuff %d, got %zd\n"
223 ,
224 cchPathIn, cchPath,
225 s_aTests[i].cchDir, cchDir,
226 s_aTests[i].offName, offName,
227 (s_aTests[i].offSuffix >= cchPathIn ? -1 : s_aTests[i].offSuffix), offSuff);
228 }
229 }
230}
231
232
233static void testParentLength(RTTEST hTest)
234{
235 static struct
236 {
237 const char *pszPath;
238 uint32_t cchNonParent;
239 uint32_t fFlags;
240 } const s_aTests[] =
241 {
242 { "/usr/bin", 3, RTPATH_STR_F_STYLE_UNIX },
243 { "/usr/bin", 3, RTPATH_STR_F_STYLE_DOS },
244 { "\\usr\\bin", 3, RTPATH_STR_F_STYLE_DOS },
245 { "/usr/bin/", 4, RTPATH_STR_F_STYLE_UNIX },
246 { "/usr/bin/", 4, RTPATH_STR_F_STYLE_DOS },
247 { "\\usr\\bin\\", 4, RTPATH_STR_F_STYLE_DOS },
248 { "A:\\usr\\bin\\", 4, RTPATH_STR_F_STYLE_DOS },
249 { "/bin", 3, RTPATH_STR_F_STYLE_UNIX },
250 { "/bin", 3, RTPATH_STR_F_STYLE_DOS },
251 { "\\bin", 3, RTPATH_STR_F_STYLE_DOS },
252 { "A:\\bin", 3, RTPATH_STR_F_STYLE_DOS },
253 { "A:/bin", 3, RTPATH_STR_F_STYLE_DOS },
254 { "A:bin", 3, RTPATH_STR_F_STYLE_DOS },
255 { "/bin/", 4, RTPATH_STR_F_STYLE_UNIX },
256 { "/bin/", 4, RTPATH_STR_F_STYLE_DOS },
257 { "A:\\bin\\", 4, RTPATH_STR_F_STYLE_DOS },
258 { "A:/bin\\", 4, RTPATH_STR_F_STYLE_DOS },
259 { "A:bin\\", 4, RTPATH_STR_F_STYLE_DOS },
260 { "/", 0, RTPATH_STR_F_STYLE_UNIX },
261 { "/", 0, RTPATH_STR_F_STYLE_DOS },
262 { "\\", 0, RTPATH_STR_F_STYLE_DOS },
263 { "A:\\", 0, RTPATH_STR_F_STYLE_DOS },
264 { "A:", 0, RTPATH_STR_F_STYLE_DOS },
265 { "bin", 3, RTPATH_STR_F_STYLE_UNIX },
266 { "bin", 3, RTPATH_STR_F_STYLE_DOS },
267 { "//unc/bin/bin", 3, RTPATH_STR_F_STYLE_DOS },
268 { "//unc/bin/bin/", 4, RTPATH_STR_F_STYLE_DOS },
269 { "//unc/bin", 3, RTPATH_STR_F_STYLE_DOS },
270 { "//unc/bin/", 4, RTPATH_STR_F_STYLE_DOS },
271 { "//unc/", 0, RTPATH_STR_F_STYLE_DOS },
272 { "//unc", 0, RTPATH_STR_F_STYLE_DOS },
273 };
274
275 RTTestSub(hTest, "RTPathParentLength");
276 for (uint32_t i = 0; i < RT_ELEMENTS(s_aTests); i++)
277 {
278 size_t const cchParent = RTPathParentLengthEx(s_aTests[i].pszPath, s_aTests[i].fFlags);
279 size_t const cchExpected = strlen(s_aTests[i].pszPath) - s_aTests[i].cchNonParent;
280 if (cchParent != cchExpected)
281 RTTestFailed(hTest, "sub-test #%u: got %u, expected %u (%s)",
282 i, cchParent, cchExpected, s_aTests[i].pszPath);
283 if (s_aTests[i].fFlags == RTPATH_STYLE)
284 {
285 size_t const cchParent2 = RTPathParentLength(s_aTests[i].pszPath);
286 if (cchParent2 != cchExpected)
287 RTTestFailed(hTest, "sub-test #%u: RTPathParentLength returned %u, expected %u (%s)",
288 i, cchParent2, cchExpected, s_aTests[i].pszPath);
289 }
290 }
291}
292
293
294static void testPurgeFilename(RTTEST hTest)
295{
296 static struct
297 {
298 const char *pszIn, *pszOut;
299 uint32_t fFlags;
300 } const s_aTests[] =
301 {
302 { "start///end", "start___end", RTPATH_STR_F_STYLE_UNIX },
303 { "start///end", "start___end", RTPATH_STR_F_STYLE_DOS },
304 { "start///end", "start___end", RTPATH_STR_F_STYLE_HOST },
305 { "1:<>\\9", "1:<>\\9", RTPATH_STR_F_STYLE_UNIX },
306 { "1:<>\\9", "1____9", RTPATH_STR_F_STYLE_DOS },
307 { "\t\r\n", "\t\r\n", RTPATH_STR_F_STYLE_UNIX },
308 { "\t\r\n", "___", RTPATH_STR_F_STYLE_DOS },
309 };
310 RTTestSub(hTest, "RTPathPurgeFilename");
311 for (uint32_t i = 0; i < RT_ELEMENTS(s_aTests); i++)
312 {
313 char szPath[RTPATH_MAX];
314 strcpy(szPath, s_aTests[i].pszIn);
315 char *pszRet = RTPathPurgeFilename(szPath, s_aTests[i].fFlags);
316 RTTEST_CHECK(hTest, pszRet == &szPath[0]);
317 if (strcmp(szPath, s_aTests[i].pszOut) != 0)
318 RTTestFailed(hTest, "sub-test #%u: got '%s', expected '%s' (style %#x)",
319 i, szPath, s_aTests[i].pszOut, s_aTests[i].fFlags);
320 }
321}
322
323
324static void testEnsureTrailingSeparator(RTTEST hTest)
325{
326 static struct
327 {
328 const char *pszIn, *pszOut;
329 uint32_t fFlags;
330 } const s_aTests[] =
331 {
332 { "/foo", "/foo/", RTPATH_STR_F_STYLE_UNIX },
333 { "/foo\\", "/foo\\/", RTPATH_STR_F_STYLE_UNIX },
334 { "/foo:", "/foo:/", RTPATH_STR_F_STYLE_UNIX },
335 { "/foo/", "/foo/", RTPATH_STR_F_STYLE_UNIX },
336 { "D:/foo", "D:/foo\\", RTPATH_STR_F_STYLE_DOS },
337 { "D:/foo\\", "D:/foo\\", RTPATH_STR_F_STYLE_DOS },
338 { "", "./", RTPATH_STR_F_STYLE_UNIX},
339 { "", ".\\", RTPATH_STR_F_STYLE_DOS },
340 { "", "." RTPATH_SLASH_STR, RTPATH_STR_F_STYLE_HOST },
341 { ".", "." RTPATH_SLASH_STR, RTPATH_STR_F_STYLE_HOST },
342 { "x", "x" RTPATH_SLASH_STR, RTPATH_STR_F_STYLE_HOST },
343 { "y" RTPATH_SLASH_STR, "y" RTPATH_SLASH_STR, RTPATH_STR_F_STYLE_HOST },
344 };
345 RTTestSub(hTest, "RTPathEnsureTrailingSeparatorEx");
346 for (uint32_t i = 0; i < RT_ELEMENTS(s_aTests); i++)
347 {
348 char szPath[RTPATH_MAX];
349 strcpy(szPath, s_aTests[i].pszIn);
350 size_t cchRet = RTPathEnsureTrailingSeparatorEx(szPath, sizeof(szPath), s_aTests[i].fFlags);
351 RTTEST_CHECK(hTest, cchRet == strlen(s_aTests[i].pszOut));
352 if (strcmp(szPath, s_aTests[i].pszOut) != 0)
353 RTTestFailed(hTest, "sub-test #%u: got '%s', expected '%s' (style %#x)",
354 i, szPath, s_aTests[i].pszOut, s_aTests[i].fFlags);
355 }
356}
357
358
359static void testFindCommon(RTTEST hTest)
360{
361 RTTestSub(hTest, "RTPathFindCommon");
362
363 static struct
364 {
365 char const *apszPaths[4];
366 uint32_t fFlags;
367 char const *pszCommon;
368 } const aTests[] =
369 {
370 /* Simple stuff first. */
371 { { "", "", "", NULL, }, RTPATH_STR_F_STYLE_UNIX,
372 "" },
373 { { "", "", "", NULL, }, RTPATH_STR_F_STYLE_DOS,
374 "" },
375 { { "none", "none", "", NULL, }, RTPATH_STR_F_STYLE_UNIX,
376 "" },
377 { { "none", "none", "", NULL, }, RTPATH_STR_F_STYLE_DOS,
378 "" },
379 { { "same", "same", "same", "same", }, RTPATH_STR_F_STYLE_UNIX,
380 "same" },
381 { { "same", "same", "same", "same", }, RTPATH_STR_F_STYLE_DOS,
382 "same" },
383 /* More complicated. */
384 { { "/path/to/stuff1", "path/to/stuff2", NULL, NULL, }, RTPATH_STR_F_STYLE_UNIX,
385 "" },
386 { { "/path/to/stuff1", "/path/to/stuff2", "/path/to/stuff3", NULL, }, RTPATH_STR_F_STYLE_UNIX,
387 "/path/to/" },
388 { { "/path/to/stuff1", "/path/to/", "/path/", NULL, }, RTPATH_STR_F_STYLE_UNIX,
389 "/path/" },
390 { { "/path/to/stuff1", "/", "/path/", NULL, }, RTPATH_STR_F_STYLE_UNIX,
391 "/" },
392 { { "/path/to/../stuff1", "./../", "/path/to/stuff2/..", NULL, }, RTPATH_STR_F_STYLE_UNIX,
393 "" },
394 { { "a/single/path", NULL, NULL, NULL, }, RTPATH_STR_F_STYLE_UNIX,
395 "a/single/path" },
396 { { "a/single\\path", NULL, NULL, NULL, }, RTPATH_STR_F_STYLE_DOS,
397 "a/single\\path" },
398 { { "C:\\Windows", NULL, NULL, NULL, }, RTPATH_STR_F_STYLE_DOS,
399 "C:\\Windows" },
400 { { "c:/windows", "c:\\program files", "C:\\AppData", NULL, }, RTPATH_STR_F_STYLE_DOS,
401 "c:/" },
402 { { "c:/windows", "c:windows", "C:system32", NULL, }, RTPATH_STR_F_STYLE_DOS,
403 "c:" },
404 { { "c:/windows", "d:windows", "e:windows", NULL, }, RTPATH_STR_F_STYLE_DOS,
405 "" },
406 { { "//usr/bin/env", "/usr//bin/env", "/usr/bin///env", "/usr/bin/env", }, RTPATH_STR_F_STYLE_UNIX,
407 "//usr/bin/env" },
408 { { "//usr/bin/env", "/usr//./././bin/env", "/usr/bin///env", "/usr/bin/env", }, RTPATH_STR_F_STYLE_UNIX,
409 "//usr/bin/env" },
410 { { "//./what/ever", "\\\\.\\what\\is\\up", "\\\\.\\\\what\\is\\up", NULL, }, RTPATH_STR_F_STYLE_DOS,
411 "//./what/" },
412 { { "//./unc/is/weird", "///./unc/is/weird", NULL, NULL, }, RTPATH_STR_F_STYLE_DOS,
413 "" },
414 { { "//system360/share", "//system370/share", "//system390/share", NULL, }, RTPATH_STR_F_STYLE_DOS,
415 "" },
416 { { "//system370/share1", "//sysTEM370/share2", "//SYsTeM370/share3", NULL, }, RTPATH_STR_F_STYLE_DOS,
417 "//system370/" },
418 { { "//system370/share1", "Z:/", NULL, NULL, }, RTPATH_STR_F_STYLE_DOS,
419 "" },
420 { { "//system370/share1", "/", NULL, NULL, }, RTPATH_STR_F_STYLE_DOS,
421 "" },
422 { { "//system370/share1", "somedir", NULL, NULL, }, RTPATH_STR_F_STYLE_DOS,
423 "" },
424 { { "/path/to/stuff1", "path/to/stuff2", NULL, NULL, }, RTPATH_STR_F_STYLE_UNIX | RTPATH_STR_F_NO_START,
425 "/path/to/" },
426 { { "path/to/stuff1", "//path\\/to\\stuff2", NULL, NULL, }, RTPATH_STR_F_STYLE_DOS | RTPATH_STR_F_NO_START,
427 "path/to/" },
428 /* '..' elements are not supported for now and leads to zero return, unless RTPATHFINDCOMMON_F_IGNORE_DOTDOT is given. */
429 { { "/usr/bin/env", "/usr/../usr/bin/env", "/usr/bin/../bin/env", NULL, }, RTPATH_STR_F_STYLE_UNIX,
430 "" },
431 { { "/lib/", "/lib/amd64/../lib.so", "/lib/i386/../libdl.so", NULL, }, RTPATH_STR_F_STYLE_UNIX,
432 "" },
433 { { "/lib/", "/lib/amd64/../lib.so", "/lib/i386/../libdl.so", NULL, }, RTPATH_STR_F_STYLE_UNIX | RTPATHFINDCOMMON_F_IGNORE_DOTDOT,
434 "/lib/" },
435 };
436
437 for (size_t i = 0; i < RT_ELEMENTS(aTests); i++)
438 {
439 size_t cPaths = RT_ELEMENTS(aTests[i].apszPaths);
440 while (cPaths > 0 && aTests[i].apszPaths[cPaths - 1] == NULL)
441 cPaths--;
442
443 size_t const cchCommon = RTPathFindCommonEx(cPaths, aTests[i].apszPaths, aTests[i].fFlags);
444 size_t const cchExpect = strlen(aTests[i].pszCommon);
445 if (cchCommon != cchExpect)
446 RTTestFailed(hTest,
447 "Test %zu failed: got %zu, expected %zu (cPaths=%zu: '%s' '%s' '%s' '%s', fFlags=%#x)", i, cchCommon,
448 cchExpect, cPaths, aTests[i].apszPaths[0], aTests[i].apszPaths[1], aTests[i].apszPaths[2],
449 aTests[i].apszPaths[3], aTests[i].fFlags);
450 }
451}
452
453
454int main()
455{
456 char szPath[RTPATH_MAX];
457
458 /*
459 * Init RT+Test.
460 */
461 RTTEST hTest;
462 int rc = RTTestInitAndCreate("tstRTPath", &hTest);
463 if (rc)
464 return rc;
465 RTTestBanner(hTest);
466
467 RTTestSub(hTest, "Environment");
468#if defined(RT_OS_OS2) || defined(RT_OS_WINDOWS)
469 RTTESTI_CHECK(RTPATH_STYLE == RTPATH_STR_F_STYLE_DOS);
470# if RTPATH_STYLE == RTPATH_STR_F_STYLE_DOS
471# else
472 RTTestIFailed("#if RTPATH_STYLE == RTPATH_STR_F_STYLE_DOS");
473# endif
474 RTTESTI_CHECK(strcmp(RTPATH_SLASH_STR, "\\") == 0);
475 RTTESTI_CHECK(RTPATH_SLASH == '\\');
476 RTTESTI_CHECK(RTPATH_IS_SEP('/'));
477 RTTESTI_CHECK(RTPATH_IS_SEP('\\'));
478 RTTESTI_CHECK(RTPATH_IS_SEP(':'));
479
480#else
481 RTTESTI_CHECK(RTPATH_STYLE == RTPATH_STR_F_STYLE_UNIX);
482# if RTPATH_STYLE == RTPATH_STR_F_STYLE_UNIX
483# else
484 RTTestIFailed("#if RTPATH_STYLE == RTPATH_STR_F_STYLE_UNIX");
485# endif
486 RTTESTI_CHECK(strcmp(RTPATH_SLASH_STR, "/") == 0);
487 RTTESTI_CHECK(RTPATH_SLASH == '/');
488 RTTESTI_CHECK(RTPATH_IS_SEP('/'));
489 RTTESTI_CHECK(!RTPATH_IS_SEP('\\'));
490 RTTESTI_CHECK(!RTPATH_IS_SEP(':'));
491#endif
492
493 /*
494 * RTPathExecDir, RTPathUserHome and RTProcGetExecutablePath.
495 */
496 RTTestSub(hTest, "RTPathExecDir");
497 RTTESTI_CHECK_RC(rc = RTPathExecDir(szPath, sizeof(szPath)), VINF_SUCCESS);
498 if (RT_SUCCESS(rc))
499 RTTestIPrintf(RTTESTLVL_INFO, "ExecDir={%s}\n", szPath);
500
501 RTTestSub(hTest, "RTProcGetExecutablePath");
502 if (RTProcGetExecutablePath(szPath, sizeof(szPath)) == szPath)
503 RTTestIPrintf(RTTESTLVL_INFO, "ExecutableName={%s}\n", szPath);
504 else
505 RTTestIFailed("RTProcGetExecutablePath -> NULL");
506
507 RTTestSub(hTest, "RTPathUserHome");
508 RTTESTI_CHECK_RC(rc = RTPathUserHome(szPath, sizeof(szPath)), VINF_SUCCESS);
509 if (RT_SUCCESS(rc))
510 RTTestIPrintf(RTTESTLVL_INFO, "UserHome={%s}\n", szPath);
511
512 RTTestSub(hTest, "RTPathUserDocuments");
513 RTTESTI_CHECK_RC(rc = RTPathUserDocuments(szPath, sizeof(szPath)), VINF_SUCCESS);
514 if (RT_SUCCESS(rc))
515 RTTestIPrintf(RTTESTLVL_INFO, "UserDocuments={%s}\n", szPath);
516
517 RTTestSub(hTest, "RTPathTemp");
518 RTTESTI_CHECK_RC(rc = RTPathTemp(szPath, sizeof(szPath)), VINF_SUCCESS);
519 if (RT_SUCCESS(rc))
520 RTTestIPrintf(RTTESTLVL_INFO, "PathTemp={%s}\n", szPath);
521 size_t cch = strlen(szPath);
522 RTTESTI_CHECK_RC(RTPathTemp(szPath, cch), VERR_BUFFER_OVERFLOW);
523 RTTESTI_CHECK_RC(RTPathTemp(szPath, cch+1), VINF_SUCCESS);
524 RTTESTI_CHECK_RC(RTPathTemp(szPath, cch+2), VINF_SUCCESS);
525
526
527 /*
528 * RTPathAbsEx.
529 */
530 RTTestSub(hTest, "RTPathAbsEx");
531 static const struct
532 {
533 uint32_t fFlags;
534 const char *pcszInputBase;
535 const char *pcszInputPath;
536 int rc;
537 const char *pcszOutput;
538 }
539 s_aRTPathAbsExTests[] =
540 {
541 { RTPATH_STR_F_STYLE_HOST, NULL, "", VERR_PATH_ZERO_LENGTH, NULL },
542 { RTPATH_STR_F_STYLE_HOST, NULL, ".", VINF_SUCCESS, "%p" },
543#if defined (RT_OS_OS2) || defined (RT_OS_WINDOWS)
544 { RTPATH_STR_F_STYLE_DOS, NULL, "\\", VINF_SUCCESS, "%d\\" },
545 { RTPATH_STR_F_STYLE_DOS, NULL, "\\..", VINF_SUCCESS, "%d\\" },
546 { RTPATH_STR_F_STYLE_DOS, NULL, "/absolute/..", VINF_SUCCESS, "%d\\" },
547 { RTPATH_STR_F_STYLE_DOS, NULL, "/absolute\\\\../..", VINF_SUCCESS, "%d\\" },
548 { RTPATH_STR_F_STYLE_DOS, NULL, "/absolute//../path\\", VINF_SUCCESS, "%d\\path\\" },
549 { RTPATH_STR_F_STYLE_DOS, NULL, "/absolute/../../path", VINF_SUCCESS, "%d\\path" },
550 { RTPATH_STR_F_STYLE_DOS, NULL, "relative/../dir\\.\\.\\.\\file.txt", VINF_SUCCESS, "%p\\dir\\file.txt" },
551 { RTPATH_STR_F_STYLE_DOS, NULL, "\\data\\", VINF_SUCCESS, "%d\\data\\" },
552 { RTPATH_STR_F_STYLE_DOS, "relative_base/dir\\", "\\from_root", VINF_SUCCESS, "%d\\from_root" },
553 { RTPATH_STR_F_STYLE_DOS, "relative_base/dir/", "relative_also", VINF_SUCCESS, "%p\\relative_base\\dir\\relative_also" },
554#else
555 { RTPATH_STR_F_STYLE_UNIX, NULL, ".", VINF_SUCCESS, "%p" },
556 { RTPATH_STR_F_STYLE_UNIX, NULL, "relative/../dir/./././file.txt", VINF_SUCCESS, "%p/dir/file.txt" },
557 { RTPATH_STR_F_STYLE_UNIX, NULL, "relative/../dir\\.\\.\\.\\file.txt", VINF_SUCCESS, "%p/dir\\.\\.\\.\\file.txt" }, /* linux-specific */
558 { RTPATH_STR_F_STYLE_UNIX, "relative_base/dir/", "/from_root", VINF_SUCCESS, "/from_root" },
559 { RTPATH_STR_F_STYLE_UNIX, "relative_base/dir/", "relative_also", VINF_SUCCESS, "%p/relative_base/dir/relative_also" },
560#endif
561 { RTPATH_STR_F_STYLE_UNIX, NULL, "/", VINF_SUCCESS, "/" },
562 { RTPATH_STR_F_STYLE_UNIX, NULL, "/..", VINF_SUCCESS, "/" },
563 { RTPATH_STR_F_STYLE_UNIX, NULL, "/absolute/..", VINF_SUCCESS, "/" },
564 { RTPATH_STR_F_STYLE_UNIX, NULL, "/absolute\\\\../..", VINF_SUCCESS, "/" },
565 { RTPATH_STR_F_STYLE_UNIX, NULL, "/absolute//../path/", VINF_SUCCESS, "/path/" },
566 { RTPATH_STR_F_STYLE_UNIX, NULL, "/absolute/../../path", VINF_SUCCESS, "/path" },
567 { RTPATH_STR_F_STYLE_UNIX, NULL, "/data/", VINF_SUCCESS, "/data/" },
568#if defined (RT_OS_OS2) || defined (RT_OS_WINDOWS)
569 { RTPATH_STR_F_STYLE_DOS, NULL, "C:\\", VINF_SUCCESS, "C:\\" },
570 { RTPATH_STR_F_STYLE_DOS, "C:\\", "..", VINF_SUCCESS, "C:\\" },
571 { RTPATH_STR_F_STYLE_DOS, "C:\\temp", "..", VINF_SUCCESS, "C:\\" },
572 { RTPATH_STR_F_STYLE_DOS, "C:\\VirtualBox/Machines", "..\\VirtualBox.xml", VINF_SUCCESS, "C:\\VirtualBox\\VirtualBox.xml" },
573 { RTPATH_STR_F_STYLE_DOS, "C:\\MustDie", "\\from_root/dir/..", VINF_SUCCESS, "C:\\from_root" },
574 { RTPATH_STR_F_STYLE_DOS, "C:\\temp", "D:\\data", VINF_SUCCESS, "D:\\data" },
575 { RTPATH_STR_F_STYLE_DOS, NULL, "\\\\server\\..\\share", VINF_SUCCESS, "\\\\server\\..\\share" /* kind of strange */ },
576 { RTPATH_STR_F_STYLE_DOS, NULL, "\\\\server/", VINF_SUCCESS, "\\\\server\\" },
577 { RTPATH_STR_F_STYLE_DOS, NULL, "\\\\", VINF_SUCCESS, "\\\\" },
578 { RTPATH_STR_F_STYLE_DOS, NULL, "\\\\\\something", VINF_SUCCESS, "\\\\\\something" /* kind of strange */ },
579 { RTPATH_STR_F_STYLE_DOS, "\\\\server\\share_as_base", "/from_root", VINF_SUCCESS, "\\\\server\\share_as_base\\from_root" },
580 { RTPATH_STR_F_STYLE_DOS, "\\\\just_server", "/from_root", VINF_SUCCESS, "\\\\just_server\\from_root" },
581 { RTPATH_STR_F_STYLE_DOS, "\\\\server\\share_as_base", "relative\\data", VINF_SUCCESS, "\\\\server\\share_as_base\\relative\\data" },
582 { RTPATH_STR_F_STYLE_DOS, "base", "\\\\?\\UNC\\relative/edwef/..", VINF_SUCCESS, "\\\\?\\UNC\\relative" },
583 { RTPATH_STR_F_STYLE_DOS, "\\\\?\\UNC\\base", "/from_root", VINF_SUCCESS, "\\\\?\\from_root" },
584 { RTPATH_STR_F_STYLE_DOS, "\\\\?\\UNC\\base", "./..", VINF_SUCCESS, "\\\\?\\UNC" },
585 { RTPATH_STR_F_STYLE_DOS | RTPATHABS_F_STOP_AT_BASE, "\\\\?\\UNC\\base", "./..", VINF_SUCCESS, "\\\\?\\UNC\\base" },
586 { RTPATH_STR_F_STYLE_DOS | RTPATHABS_F_STOP_AT_BASE, "\\\\?\\UNC\\base", "/..", VINF_SUCCESS, "\\\\?\\" },
587 { RTPATH_STR_F_STYLE_DOS, NULL, "\\\\.\\asdf\\..", VINF_SUCCESS, "\\\\.\\" },
588 { RTPATH_STR_F_STYLE_DOS, NULL, "\\\\?\\asdf\\..", VINF_SUCCESS, "\\\\?\\" },
589 { RTPATH_STR_F_STYLE_DOS, NULL, "\\\\x\\asdf\\..", VINF_SUCCESS, "\\\\x\\asdf" },
590#else
591 { RTPATH_STR_F_STYLE_UNIX, "\\temp", "\\data", VINF_SUCCESS, "%p/\\temp/\\data" },
592#endif
593 { RTPATH_STR_F_STYLE_UNIX, "/VirtualBox/Machines", "../VirtualBox.xml", VINF_SUCCESS, "/VirtualBox/VirtualBox.xml" },
594 { RTPATH_STR_F_STYLE_UNIX, "/MustDie", "/from_root/dir/..", VINF_SUCCESS, "/from_root" },
595 { RTPATH_STR_F_STYLE_UNIX, "/temp", "..", VINF_SUCCESS, "/" },
596 };
597
598 char *pszGuardedBuf = NULL;
599 rc = RTTestGuardedAlloc(hTest, RTPATH_MAX, 0, false /*fHead*/, (void **)&pszGuardedBuf);
600 if (RT_FAILURE(rc))
601 pszGuardedBuf = szPath;
602
603 for (unsigned i = 0; i < RT_ELEMENTS(s_aRTPathAbsExTests); ++ i)
604 {
605 if (RT_FAILURE(s_aRTPathAbsExTests[i].rc))
606 RTTestDisableAssertions(hTest);
607
608 size_t cbAbsPath = sizeof(szPath);
609 rc = RTPathAbsEx(s_aRTPathAbsExTests[i].pcszInputBase,
610 s_aRTPathAbsExTests[i].pcszInputPath,
611 s_aRTPathAbsExTests[i].fFlags,
612 szPath, &cbAbsPath);
613
614 if (RT_FAILURE(s_aRTPathAbsExTests[i].rc))
615 RTTestRestoreAssertions(hTest);
616
617 if (rc != s_aRTPathAbsExTests[i].rc)
618 {
619 RTTestIFailed("#%u: unexpected result code!\n"
620 " flags: %#x\n"
621 " input base: '%s'\n"
622 " input path: '%s'\n"
623 " output: '%s'\n"
624 " rc: %Rrc\n"
625 " expected rc: %Rrc",
626 i,
627 s_aRTPathAbsExTests[i].fFlags,
628 s_aRTPathAbsExTests[i].pcszInputBase,
629 s_aRTPathAbsExTests[i].pcszInputPath,
630 szPath, rc,
631 s_aRTPathAbsExTests[i].rc);
632 continue;
633 }
634
635 char szTmp[RTPATH_MAX];
636 char *pszExpected = NULL;
637 if (s_aRTPathAbsExTests[i].pcszOutput != NULL)
638 {
639 if (s_aRTPathAbsExTests[i].pcszOutput[0] == '%')
640 {
641 RTTESTI_CHECK_RC(rc = RTPathGetCurrent(szTmp, sizeof(szTmp)), VINF_SUCCESS);
642 if (RT_FAILURE(rc))
643 break;
644
645 pszExpected = szTmp;
646
647 if (s_aRTPathAbsExTests[i].pcszOutput[1] == 'p')
648 {
649 cch = strlen(szTmp);
650 if (cch + strlen(s_aRTPathAbsExTests[i].pcszOutput) - 2 <= sizeof(szTmp))
651 strcpy(szTmp + cch, s_aRTPathAbsExTests[i].pcszOutput + 2);
652 }
653#if defined(RT_OS_OS2) || defined(RT_OS_WINDOWS)
654 else if (s_aRTPathAbsExTests[i].pcszOutput[1] == 'd')
655 {
656 if (2 + strlen(s_aRTPathAbsExTests[i].pcszOutput) - 2 <= sizeof(szTmp))
657 strcpy(szTmp + 2, s_aRTPathAbsExTests[i].pcszOutput + 2);
658 }
659#endif
660 }
661 else
662 {
663 strcpy(szTmp, s_aRTPathAbsExTests[i].pcszOutput);
664 pszExpected = szTmp;
665 }
666
667 if ( strcmp(szPath, pszExpected)
668 || strlen(szPath) != cbAbsPath)
669 {
670 RTTestIFailed("#%u: Unexpected result\n"
671 " flags: %#x\n"
672 " input base: '%s'\n"
673 " input path: '%s'\n"
674 " output: '%s'\n"
675 " expected: '%s' ('%s')\n"
676 " cchResult: %#x, actual %#x",
677 i,
678 s_aRTPathAbsExTests[i].fFlags,
679 s_aRTPathAbsExTests[i].pcszInputBase,
680 s_aRTPathAbsExTests[i].pcszInputPath,
681 szPath,
682 pszExpected, s_aRTPathAbsExTests[i].pcszOutput,
683 cbAbsPath, strlen(szPath));
684 continue;
685 }
686
687 if (RT_SUCCESS(s_aRTPathAbsExTests[i].rc))
688 {
689 /* Test the RTPATHABS_F_ENSURE_TRAILING_SLASH flag: */
690 cbAbsPath = sizeof(szPath);
691 rc = RTPathAbsEx(s_aRTPathAbsExTests[i].pcszInputBase,
692 s_aRTPathAbsExTests[i].pcszInputPath,
693 s_aRTPathAbsExTests[i].fFlags | RTPATHABS_F_ENSURE_TRAILING_SLASH,
694 szPath, &cbAbsPath);
695 char chSlash = (s_aRTPathAbsExTests[i].fFlags & RTPATH_STR_F_STYLE_MASK) == RTPATH_STR_F_STYLE_DOS ? '\\'
696 : (s_aRTPathAbsExTests[i].fFlags & RTPATH_STR_F_STYLE_MASK) == RTPATH_STR_F_STYLE_UNIX ? '/'
697 : RTPATH_SLASH;
698 if ( RT_FAILURE(rc)
699 || strlen(szPath) != cbAbsPath
700 || szPath[cbAbsPath - 1] != chSlash)
701 RTTestIFailed("#%u: Unexpected RTPATHABS_F_ENSURE_TRAILING_SLASH result: %Rrc\n"
702 " flags: %#x | RTPATHABS_F_ENSURE_TRAILING_SLASH\n"
703 " input base: '%s'\n"
704 " input path: '%s'\n"
705 " output: '%s' ('%c' vs '%c')\n"
706 " cchResult: %#x, actual %#x",
707 i, rc,
708 s_aRTPathAbsExTests[i].fFlags,
709 s_aRTPathAbsExTests[i].pcszInputBase,
710 s_aRTPathAbsExTests[i].pcszInputPath,
711 szPath, szPath[cbAbsPath - 1], chSlash,
712 cbAbsPath, strlen(szPath));
713
714 /* Do overflow testing: */
715 size_t const cbNeeded = strlen(pszExpected) + 1;
716 for (size_t cbBuf = 0; cbBuf < cbNeeded + 64; cbBuf++)
717 {
718 char *pszBuf = &pszGuardedBuf[RTPATH_MAX - cbBuf];
719 memset(pszBuf, 0x33, cbBuf);
720 cbAbsPath = cbBuf;
721 rc = RTPathAbsEx(s_aRTPathAbsExTests[i].pcszInputBase, s_aRTPathAbsExTests[i].pcszInputPath,
722 s_aRTPathAbsExTests[i].fFlags, pszBuf, &cbAbsPath);
723 if ( cbBuf < cbNeeded
724 && ( rc != VERR_BUFFER_OVERFLOW
725 || cbAbsPath < cbNeeded))
726 RTTestIFailed("#%u: Unexpected overflow result: %Rrc%s\n"
727 " flags: %#x\n"
728 " input base: '%s'\n"
729 " input path: '%s'\n"
730 " cbBuf[in]: %#x\n"
731 " cbBuf[out]: %#x\n"
732 " cbNeeded: %#x\n",
733 i, rc, rc != VERR_BUFFER_OVERFLOW ? " - expected VERR_BUFFER_OVERFLOW" : "",
734 s_aRTPathAbsExTests[i].fFlags,
735 s_aRTPathAbsExTests[i].pcszInputBase,
736 s_aRTPathAbsExTests[i].pcszInputPath,
737 cbBuf,
738 cbAbsPath,
739 cbNeeded);
740 else if ( cbBuf >= cbNeeded
741 && ( rc != s_aRTPathAbsExTests[i].rc
742 || cbAbsPath != cbNeeded - 1
743 || strcmp(pszBuf, pszExpected)
744 || strlen(pszBuf) != cbAbsPath))
745 RTTestIFailed("#%u: Unexpected result: %Rrc (expected %Rrc)\n"
746 " flags: %#x\n"
747 " input base: '%s'\n"
748 " input path: '%s'\n"
749 " cbBuf[in]: %#x\n"
750 " cbBuf[out]: %#x\n"
751 " cbNeeded: %#x\n",
752 i, rc, s_aRTPathAbsExTests[i].rc,
753 s_aRTPathAbsExTests[i].fFlags,
754 s_aRTPathAbsExTests[i].pcszInputBase,
755 s_aRTPathAbsExTests[i].pcszInputPath,
756 cbBuf,
757 cbAbsPath,
758 cbNeeded);
759
760 }
761 }
762
763 /* RTPathAbsExDup */
764 char *pszDup = RTPathAbsExDup(s_aRTPathAbsExTests[i].pcszInputBase,
765 s_aRTPathAbsExTests[i].pcszInputPath,
766 s_aRTPathAbsExTests[i].fFlags);
767 if ( (RT_SUCCESS(s_aRTPathAbsExTests[i].rc) ? pszDup == NULL : pszDup != NULL)
768 || RTStrCmp(pszDup, pszExpected))
769 RTTestIFailed("#%u: Unexpected RTPathAbsExDup result: %p%s\n"
770 " flags: %#x\n"
771 " input base: '%s'\n"
772 " input path: '%s'\n"
773 " output: '%s'\n"
774 " expected: '%s' ('%s')\n",
775 i, pszDup,
776 (RT_SUCCESS(s_aRTPathAbsExTests[i].rc) ? pszDup == NULL : pszDup != NULL) ? pszDup ? "NULL" : "!NULL" : "",
777 s_aRTPathAbsExTests[i].fFlags,
778 s_aRTPathAbsExTests[i].pcszInputBase,
779 s_aRTPathAbsExTests[i].pcszInputPath,
780 pszDup,
781 pszExpected, s_aRTPathAbsExTests[i].pcszOutput);
782 RTStrFree(pszDup);
783 }
784 }
785
786 if (pszGuardedBuf != szPath)
787 RTTestGuardedFree(hTest, pszGuardedBuf);
788
789
790 /*
791 * RTPathStripFilename
792 */
793 RTTestSub(hTest, "RTPathStripFilename");
794 static const char *s_apszStripFilenameTests[] =
795 {
796 "/usr/include///", "/usr/include//",
797 "/usr/include/", "/usr/include",
798 "/usr/include", "/usr",
799 "/usr", "/",
800 "usr", ".",
801#if defined (RT_OS_OS2) || defined (RT_OS_WINDOWS)
802 "c:/windows", "c:/",
803 "c:/", "c:/",
804 "D:", "D:",
805 "C:\\OS2\\DLLS", "C:\\OS2",
806#endif
807 };
808 for (unsigned i = 0; i < RT_ELEMENTS(s_apszStripFilenameTests); i += 2)
809 {
810 const char *pszInput = s_apszStripFilenameTests[i];
811 const char *pszExpect = s_apszStripFilenameTests[i + 1];
812 strcpy(szPath, pszInput);
813 RTPathStripFilename(szPath);
814 if (strcmp(szPath, pszExpect))
815 {
816 RTTestIFailed("Unexpected result\n"
817 " input: '%s'\n"
818 " output: '%s'\n"
819 "expected: '%s'",
820 pszInput, szPath, pszExpect);
821 }
822 }
823
824 /*
825 * RTPathAppend.
826 */
827 RTTestSub(hTest, "RTPathAppend");
828 static const char *s_apszAppendTests[] =
829 {
830 /* base append result */
831 "/", "", "/",
832 "", "/", "/",
833 "/", "/", "/",
834 "/x", "", "/x",
835 "/x", "/", "/x/",
836 "/", "x", "/x",
837 "dir", "file", "dir" RTPATH_SLASH_STR "file",
838 "dir", "/file", "dir/file",
839 "dir", "//file", "dir/file",
840 "dir", "///file", "dir/file",
841 "dir/", "/file", "dir/file",
842 "dir/", "//file", "dir/file",
843 "dir/", "///file", "dir/file",
844 "dir//", "file", "dir/file",
845 "dir//", "/file", "dir/file",
846 "dir//", "//file", "dir/file",
847 "dir///", "///file", "dir/file",
848 "/bin/testcase", "foo.r0", "/bin/testcase" RTPATH_SLASH_STR "foo.r0",
849#if defined (RT_OS_OS2) || defined (RT_OS_WINDOWS)
850 "/", "\\", "/",
851 "\\", "/", "\\",
852 "\\\\srv\\shr", "dir//", "\\\\srv\\shr" RTPATH_SLASH_STR "dir//",
853 "\\\\srv\\shr", "dir//file", "\\\\srv\\shr" RTPATH_SLASH_STR "dir//file",
854 "\\\\srv\\shr", "//dir//", "\\\\srv\\shr/dir//",
855 "\\\\srv\\shr", "/\\dir//", "\\\\srv\\shr\\dir//",
856 "\\\\", "not-srv/not-shr/file", "\\not-srv/not-shr/file",
857 "C:", "autoexec.bat", "C:autoexec.bat",
858 "C:", "/autoexec.bat", "C:/autoexec.bat",
859 "C:", "\\autoexec.bat", "C:\\autoexec.bat",
860 "C:\\", "/autoexec.bat", "C:\\autoexec.bat",
861 "C:\\\\", "autoexec.bat", "C:\\autoexec.bat",
862 "E:\\bin\\testcase", "foo.r0", "E:\\bin\\testcase" RTPATH_SLASH_STR "foo.r0",
863#endif
864 };
865 for (unsigned i = 0; i < RT_ELEMENTS(s_apszAppendTests); i += 3)
866 {
867 const char *pszInput = s_apszAppendTests[i];
868 const char *pszAppend = s_apszAppendTests[i + 1];
869 const char *pszExpect = s_apszAppendTests[i + 2];
870 strcpy(szPath, pszInput);
871 RTTESTI_CHECK_RC(rc = RTPathAppend(szPath, sizeof(szPath), pszAppend), VINF_SUCCESS);
872 if (RT_FAILURE(rc))
873 continue;
874 if (strcmp(szPath, pszExpect))
875 {
876 RTTestIFailed("Unexpected result\n"
877 " input: '%s'\n"
878 " append: '%s'\n"
879 " output: '%s'\n"
880 "expected: '%s'",
881 pszInput, pszAppend, szPath, pszExpect);
882 }
883 else
884 {
885 size_t const cchResult = strlen(szPath);
886
887 strcpy(szPath, pszInput);
888 RTTESTI_CHECK_RC(rc = RTPathAppend(szPath, cchResult + 2, pszAppend), VINF_SUCCESS);
889 RTTESTI_CHECK(RT_FAILURE(rc) || !strcmp(szPath, pszExpect));
890
891 strcpy(szPath, pszInput);
892 RTTESTI_CHECK_RC(rc = RTPathAppend(szPath, cchResult + 1, pszAppend), VINF_SUCCESS);
893 RTTESTI_CHECK(RT_FAILURE(rc) || !strcmp(szPath, pszExpect));
894
895 if (strlen(pszInput) < cchResult)
896 {
897 strcpy(szPath, pszInput);
898 RTTESTI_CHECK_RC(RTPathAppend(szPath, cchResult, pszAppend), VERR_BUFFER_OVERFLOW);
899 }
900 }
901 }
902
903 /*
904 * RTPathJoin - reuse the append tests.
905 */
906 RTTestSub(hTest, "RTPathJoin");
907 for (unsigned i = 0; i < RT_ELEMENTS(s_apszAppendTests); i += 3)
908 {
909 const char *pszInput = s_apszAppendTests[i];
910 const char *pszAppend = s_apszAppendTests[i + 1];
911 const char *pszExpect = s_apszAppendTests[i + 2];
912
913 memset(szPath, 'a', sizeof(szPath)); szPath[sizeof(szPath) - 1] = '\0';
914
915 RTTESTI_CHECK_RC(rc = RTPathJoin(szPath, sizeof(szPath), pszInput, pszAppend), VINF_SUCCESS);
916 if (RT_FAILURE(rc))
917 continue;
918 if (strcmp(szPath, pszExpect))
919 {
920 RTTestIFailed("Unexpected result\n"
921 " input: '%s'\n"
922 " append: '%s'\n"
923 " output: '%s'\n"
924 "expected: '%s'",
925 pszInput, pszAppend, szPath, pszExpect);
926 }
927 else
928 {
929 size_t const cchResult = strlen(szPath);
930
931 memset(szPath, 'a', sizeof(szPath)); szPath[sizeof(szPath) - 1] = '\0';
932 RTTESTI_CHECK_RC(rc = RTPathJoin(szPath, cchResult + 2, pszInput, pszAppend), VINF_SUCCESS);
933 RTTESTI_CHECK(RT_FAILURE(rc) || !strcmp(szPath, pszExpect));
934
935 memset(szPath, 'a', sizeof(szPath)); szPath[sizeof(szPath) - 1] = '\0';
936 RTTESTI_CHECK_RC(rc = RTPathJoin(szPath, cchResult + 1, pszInput, pszAppend), VINF_SUCCESS);
937 RTTESTI_CHECK(RT_FAILURE(rc) || !strcmp(szPath, pszExpect));
938
939 RTTESTI_CHECK_RC(rc = RTPathJoin(szPath, cchResult, pszInput, pszAppend), VERR_BUFFER_OVERFLOW);
940 }
941 }
942
943 /*
944 * RTPathJoinA - reuse the append tests.
945 */
946 RTTestSub(hTest, "RTPathJoinA");
947 for (unsigned i = 0; i < RT_ELEMENTS(s_apszAppendTests); i += 3)
948 {
949 const char *pszInput = s_apszAppendTests[i];
950 const char *pszAppend = s_apszAppendTests[i + 1];
951 const char *pszExpect = s_apszAppendTests[i + 2];
952
953 char *pszPathDst;
954 RTTESTI_CHECK(pszPathDst = RTPathJoinA(pszInput, pszAppend));
955 if (!pszPathDst)
956 continue;
957 if (strcmp(pszPathDst, pszExpect))
958 {
959 RTTestIFailed("Unexpected result\n"
960 " input: '%s'\n"
961 " append: '%s'\n"
962 " output: '%s'\n"
963 "expected: '%s'",
964 pszInput, pszAppend, pszPathDst, pszExpect);
965 }
966 RTStrFree(pszPathDst);
967 }
968
969 /*
970 * RTPathStripTrailingSlash
971 */
972 static const char *s_apszStripTrailingSlash[] =
973 {
974 /* input result */
975 "/", "/",
976 "//", "/",
977 "////////////////////", "/",
978 "/tmp", "/tmp",
979 "/tmp////////////////", "/tmp",
980 "tmp", "tmp",
981 "tmp////////////////", "tmp",
982 "./", ".",
983#if defined (RT_OS_OS2) || defined (RT_OS_WINDOWS)
984 "////////////////////", "/",
985 "D:", "D:",
986 "D:/", "D:/",
987 "D:\\", "D:\\",
988 "D:\\/\\", "D:\\",
989 "D:/\\/\\", "D:/",
990 "C:/Temp", "C:/Temp",
991 "C:/Temp/", "C:/Temp",
992 "C:/Temp\\/", "C:/Temp",
993#endif
994 };
995 for (unsigned i = 0; i < RT_ELEMENTS(s_apszStripTrailingSlash); i += 2)
996 {
997 const char *pszInput = s_apszStripTrailingSlash[i];
998 const char *pszExpect = s_apszStripTrailingSlash[i + 1];
999
1000 strcpy(szPath, pszInput);
1001 cch = RTPathStripTrailingSlash(szPath);
1002 if (strcmp(szPath, pszExpect))
1003 RTTestIFailed("Unexpected result\n"
1004 " input: '%s'\n"
1005 " output: '%s'\n"
1006 "expected: '%s'",
1007 pszInput, szPath, pszExpect);
1008 else
1009 RTTESTI_CHECK(cch == strlen(szPath));
1010 }
1011
1012 /*
1013 * RTPathCountComponents
1014 */
1015 RTTestSub(hTest, "RTPathCountComponents");
1016 RTTESTI_CHECK(RTPathCountComponents("") == 0);
1017 RTTESTI_CHECK(RTPathCountComponents("/") == 1);
1018 RTTESTI_CHECK(RTPathCountComponents("//") == 1);
1019 RTTESTI_CHECK(RTPathCountComponents("//////////////") == 1);
1020 RTTESTI_CHECK(RTPathCountComponents("//////////////bin") == 2);
1021 RTTESTI_CHECK(RTPathCountComponents("//////////////bin/") == 2);
1022 RTTESTI_CHECK(RTPathCountComponents("//////////////bin/////") == 2);
1023 RTTESTI_CHECK(RTPathCountComponents("..") == 1);
1024 RTTESTI_CHECK(RTPathCountComponents("../") == 1);
1025 RTTESTI_CHECK(RTPathCountComponents("../..") == 2);
1026 RTTESTI_CHECK(RTPathCountComponents("../../") == 2);
1027#if defined (RT_OS_OS2) || defined (RT_OS_WINDOWS)
1028 RTTESTI_CHECK(RTPathCountComponents("d:") == 1);
1029 RTTESTI_CHECK(RTPathCountComponents("d:/") == 1);
1030 RTTESTI_CHECK(RTPathCountComponents("d:/\\") == 1);
1031 RTTESTI_CHECK(RTPathCountComponents("d:\\") == 1);
1032 RTTESTI_CHECK(RTPathCountComponents("c:\\config.sys") == 2);
1033 RTTESTI_CHECK(RTPathCountComponents("c:\\windows") == 2);
1034 RTTESTI_CHECK(RTPathCountComponents("c:\\windows\\") == 2);
1035 RTTESTI_CHECK(RTPathCountComponents("c:\\windows\\system32") == 3);
1036 RTTESTI_CHECK(RTPathCountComponents("//./C$") == 1);
1037 RTTESTI_CHECK(RTPathCountComponents("\\\\.\\C$") == 1);
1038 RTTESTI_CHECK(RTPathCountComponents("/\\.\\C$") == 1);
1039 RTTESTI_CHECK(RTPathCountComponents("//myserver") == 1);
1040 RTTESTI_CHECK(RTPathCountComponents("//myserver/") == 1);
1041 RTTESTI_CHECK(RTPathCountComponents("//myserver/share") == 1);
1042 RTTESTI_CHECK(RTPathCountComponents("//myserver/share/") == 1);
1043 RTTESTI_CHECK(RTPathCountComponents("//myserver/share\\") == 1);
1044 RTTESTI_CHECK(RTPathCountComponents("//myserver/share\\x") == 2);
1045 RTTESTI_CHECK(RTPathCountComponents("//myserver/share\\x\\y") == 3);
1046 RTTESTI_CHECK(RTPathCountComponents("//myserver/share\\x\\y\\") == 3);
1047#endif
1048
1049 /*
1050 * RTPathCopyComponents
1051 */
1052 struct
1053 {
1054 const char *pszSrc;
1055 size_t cComponents;
1056 const char *pszResult;
1057 } s_aCopyComponents[] =
1058 {
1059 { "", 0, "" },
1060 { "", 5, "" },
1061 { "/", 0, "" },
1062 { "/", 1, "/" },
1063 { "/", 2, "/" },
1064 { "/usr/bin/sed", 0, "" },
1065 { "/usr/bin/sed", 1, "/" },
1066 { "/usr/bin/sed", 2, "/usr/" },
1067 { "/usr/bin/sed", 3, "/usr/bin/" },
1068 { "/usr/bin/sed", 4, "/usr/bin/sed" },
1069 { "/usr/bin/sed", 5, "/usr/bin/sed" },
1070 { "/usr/bin/sed", 6, "/usr/bin/sed" },
1071 { "/usr///bin/sed", 2, "/usr///" },
1072 };
1073 for (unsigned i = 0; i < RT_ELEMENTS(s_aCopyComponents); i++)
1074 {
1075 const char *pszInput = s_aCopyComponents[i].pszSrc;
1076 size_t cComponents = s_aCopyComponents[i].cComponents;
1077 const char *pszResult = s_aCopyComponents[i].pszResult;
1078
1079 memset(szPath, 'a', sizeof(szPath));
1080 rc = RTPathCopyComponents(szPath, sizeof(szPath), pszInput, cComponents);
1081 RTTESTI_CHECK_RC(rc, VINF_SUCCESS);
1082 if (RT_SUCCESS(rc) && strcmp(szPath, pszResult))
1083 RTTestIFailed("Unexpected result\n"
1084 " input: '%s' cComponents=%u\n"
1085 " output: '%s'\n"
1086 "expected: '%s'",
1087 pszInput, cComponents, szPath, pszResult);
1088 else if (RT_SUCCESS(rc))
1089 {
1090 RTTESTI_CHECK_RC(RTPathCopyComponents(szPath, strlen(pszResult) + 1, pszInput, cComponents), VINF_SUCCESS);
1091 RTTESTI_CHECK_RC(RTPathCopyComponents(szPath, strlen(pszResult), pszInput, cComponents), VERR_BUFFER_OVERFLOW);
1092 }
1093 }
1094
1095
1096 /*
1097 * RTPathStripSuffix
1098 */
1099 RTTestSub(hTest, "RTPathStripSuffix");
1100 struct
1101 {
1102 const char *pszSrc;
1103 const char *pszResult;
1104 } s_aStripExt[] =
1105 {
1106 { "filename.ext", "filename" },
1107 { "filename.ext1.ext2.ext3", "filename.ext1.ext2" },
1108 { "filename..ext", "filename." },
1109 { "filename.ext.", "filename.ext." },
1110 };
1111 for (unsigned i = 0; i < RT_ELEMENTS(s_aStripExt); i++)
1112 {
1113 const char *pszInput = s_aStripExt[i].pszSrc;
1114 const char *pszResult = s_aStripExt[i].pszResult;
1115
1116 strcpy(szPath, pszInput);
1117 RTPathStripSuffix(szPath);
1118 if (strcmp(szPath, pszResult))
1119 RTTestIFailed("Unexpected result\n"
1120 " input: '%s'\n"
1121 " output: '%s'\n"
1122 "expected: '%s'",
1123 pszInput, szPath, pszResult);
1124 }
1125
1126 /*
1127 * RTPathCalcRelative
1128 */
1129 RTTestSub(hTest, "RTPathCalcRelative");
1130 struct
1131 {
1132 const char *pszFrom;
1133 bool fFromFile;
1134 const char *pszTo;
1135 int rc;
1136 const char *pszExpected;
1137 } s_aRelPath[] =
1138 {
1139 { "/home/test.ext", true, "/home/test2.ext", VINF_SUCCESS, "test2.ext" },
1140 { "/dir/test.ext", true, "/dir/dir2/test2.ext", VINF_SUCCESS, "dir2/test2.ext" },
1141 { "/dir/dir2/test.ext", true, "/dir/test2.ext", VINF_SUCCESS, ".." RTPATH_SLASH_STR "test2.ext" },
1142 { "/dir/dir2/test.ext", true, "/dir/dir3/test2.ext", VINF_SUCCESS, ".." RTPATH_SLASH_STR "dir3/test2.ext" },
1143 { "/dir/dir2", false, "/dir/dir3/test2.ext", VINF_SUCCESS, ".." RTPATH_SLASH_STR "dir3/test2.ext" },
1144 { "/dir/dir2", false, "/dir/dir3//test2.ext", VINF_SUCCESS, ".." RTPATH_SLASH_STR "dir3//test2.ext" },
1145 { "/dir/dir2/", false, "/dir/dir3/test2.ext", VINF_SUCCESS, ".." RTPATH_SLASH_STR "dir3/test2.ext" },
1146 { "/dir/dir2////", false, "/dir//dir3/test2.ext", VINF_SUCCESS, ".." RTPATH_SLASH_STR "dir3/test2.ext" },
1147 { "/include/iprt", false, "/include/iprt/cdefs.h", VINF_SUCCESS, "cdefs.h" },
1148 { "/include/iprt/", false, "/include/iprt/cdefs.h", VINF_SUCCESS, "cdefs.h" },
1149 { "/include/iprt/tt.h", true, "/include/iprt/cdefs.h", VINF_SUCCESS, "cdefs.h" },
1150#if defined (RT_OS_OS2) || defined (RT_OS_WINDOWS)
1151 { "\\\\server\\share\\test.ext", true, "\\\\server\\share2\\test2.ext", VERR_NOT_SUPPORTED, "" },
1152 { "c:\\dir\\test.ext", true, "f:\\dir\\test.ext", VERR_NOT_SUPPORTED, "" },
1153 { "F:\\dir\\test.ext", false, "f:/dir//test.ext", VINF_SUCCESS, "." } ,
1154 { "F:\\diR\\Test.exT", true, "f:/dir//test.ext", VINF_SUCCESS, "Test.exT" } ,
1155 { "F:\\K\xc3\x85RE\\Test.exT", true, "f:/k\xc3\xa5re//test.ext", VINF_SUCCESS, "Test.exT" } ,
1156#endif
1157 };
1158 for (unsigned i = 0; i < RT_ELEMENTS(s_aRelPath); i++)
1159 {
1160 const char *pszFrom = s_aRelPath[i].pszFrom;
1161 bool fFromFile = s_aRelPath[i].fFromFile;
1162 const char *pszTo = s_aRelPath[i].pszTo;
1163
1164 rc = RTPathCalcRelative(szPath, sizeof(szPath), pszFrom, fFromFile, pszTo);
1165 if (rc != s_aRelPath[i].rc)
1166 RTTestIFailed("Unexpected return code for %s .. %s\n"
1167 " got: %Rrc\n"
1168 "expected: %Rrc",
1169 pszFrom, pszTo, rc, s_aRelPath[i].rc);
1170 else if ( RT_SUCCESS(rc)
1171 && strcmp(szPath, s_aRelPath[i].pszExpected))
1172 RTTestIFailed("Unexpected result\n"
1173 " from: '%s' (%s)\n"
1174 " to: '%s'\n"
1175 " output: '%s'\n"
1176 "expected: '%s'",
1177 pszFrom, fFromFile ? "file" : "dir", pszTo, szPath, s_aRelPath[i].pszExpected);
1178 }
1179
1180 testParserAndSplitter(hTest);
1181 testParentLength(hTest);
1182 testPurgeFilename(hTest);
1183 testEnsureTrailingSeparator(hTest);
1184 testFindCommon(hTest);
1185
1186 /*
1187 * Summary.
1188 */
1189 return RTTestSummaryAndDestroy(hTest);
1190}
1191
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