VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/path/RTPathFindCommon.cpp.h@ 93351

Last change on this file since 93351 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: 9.1 KB
Line 
1/* $Id: RTPathFindCommon.cpp.h 93115 2022-01-01 11:31:46Z vboxsync $ */
2/** @file
3 * IPRT - RTPathFindCommon - Code Template.
4 *
5 * This file included multiple times with different path style macros.
6 */
7
8/*
9 * Copyright (C) 2020-2022 Oracle Corporation
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 *
19 * The contents of this file may alternatively be used under the terms
20 * of the Common Development and Distribution License Version 1.0
21 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
22 * VirtualBox OSE distribution, in which case the provisions of the
23 * CDDL are applicable instead of those of the GPL.
24 *
25 * You may elect to license modified versions of this file under the
26 * terms and conditions of either the GPL or the CDDL or both.
27 */
28
29#include "rtpath-root-length-template.cpp.h"
30
31
32/** Helper for skipping slashes, given a pointer to the first one. */
33DECLINLINE(const char *) RTPATH_STYLE_FN(rtPathHlpSkipSlashes)(const char *pszSlash)
34{
35 for (;;)
36 {
37 char ch;
38 do
39 ch = *++pszSlash;
40 while (RTPATH_IS_SLASH(ch));
41
42 /* Also skip '/./' sequences. */
43 if ( ch != '.'
44 || !RTPATH_IS_SLASH(pszSlash[1]))
45 break;
46 pszSlash++;
47 }
48 return pszSlash;
49}
50
51
52static size_t RTPATH_STYLE_FN(rtPathFindCommon)(size_t cPaths, const char **papszPaths, uint32_t fFlags)
53{
54 /*
55 * Check for '..' elements before we start doing anything.
56 *
57 * They are currently not supported at all (lazy) and we shun them for
58 * security reasons. Iff we want to support them properly, we'd have to:
59 * 1. Note down exactly where the root specification ends for each of
60 * the paths so we can prevent '..' from messing with it.
61 * 2. When encountering '..', we'd have to ascend all paths.
62 * 3. When encountering a difference, we'd have to see if it's eliminated
63 * by a following '..' sequence.
64 * 4. When returning anything, we'd have to see if it could be affected by
65 * a '..' sequence later in any of the paths.
66 *
67 * We could kind of RTAbsPath the secondary paths, however it wouldn't work
68 * for the primary path we use as reference.
69 *
70 * Summa summarum: Annoyingly tedious, so just forget it.
71 */
72 if (!(fFlags & RTPATHFINDCOMMON_F_IGNORE_DOTDOT))
73 for (size_t i = 0; i < cPaths; i++)
74 {
75 const char * const psz = papszPaths[i];
76 const char *pszDot = strchr(psz, '.');
77 while (pszDot)
78 {
79 if ( pszDot[1] == '.'
80 && (RTPATH_IS_SLASH(pszDot[2]) || pszDot[2] == '\0')
81 && ( pszDot == psz
82 || RTPATH_IS_SLASH(pszDot[-1])
83#if RTPATH_STYLE == RTPATH_STR_F_STYLE_DOS
84 || (pszDot[-1] == ':' && psz + 2 == pszDot && !(fFlags & RTPATH_STR_F_NO_START))
85#endif
86 )
87 )
88 return 0;
89 pszDot = strchr(pszDot + 1, '.');
90 }
91 }
92
93 /*
94 * We use the first path as the reference for the return length.
95 */
96 const char * pszPath0 = papszPaths[0];
97 const char * pszPath0EndLastComp = pszPath0;
98 const char * const pszPath0Start = pszPath0;
99
100 /*
101 * Deal with root stuff as appropriate.
102 */
103 if (fFlags & RTPATH_STR_F_NO_START)
104 {
105 /* We ignore leading slashes when RTPATH_STR_F_NO_START is specified: */
106 for (size_t i = 0; i < cPaths; i++)
107 {
108 const char *psz = papszPaths[i];
109 papszPaths[i] = RTPATH_IS_SLASH(*psz) ? RTPATH_STYLE_FN(rtPathHlpSkipSlashes)(psz) : psz;
110 }
111 pszPath0EndLastComp = pszPath0 = papszPaths[0];
112 }
113#if RTPATH_STYLE == RTPATH_STR_F_STYLE_DOS
114 else if (RTPATH_IS_SLASH(pszPath0[0]))
115 {
116 /* UNC requires a little bit of special magic to make sure we have
117 exactly two slashes in each path and don't mix things up. */
118 char ch;
119 if ( RTPATH_IS_SLASH(pszPath0[1])
120 && (ch = pszPath0[2]) != '\0'
121 && !RTPATH_IS_SLASH(ch))
122 {
123 pszPath0 += 2;
124 for (size_t i = 1; i < cPaths; i++)
125 {
126 const char *psz = papszPaths[i];
127 if ( RTPATH_IS_SLASH(psz[0])
128 && RTPATH_IS_SLASH(psz[1])
129 && (ch = psz[2]) != '\0'
130 && !RTPATH_IS_SLASH(ch))
131 papszPaths[i] = psz + 2;
132 else
133 return 0;
134 }
135 }
136 else
137 {
138 for (size_t i = 1; i < cPaths; i++)
139 {
140 const char *psz = papszPaths[i];
141 if ( RTPATH_IS_SLASH(psz[0])
142 && RTPATH_IS_SLASH(psz[1])
143 && (ch = psz[2]) != '\0'
144 && !RTPATH_IS_SLASH(ch))
145 return 0;
146 if (!RTPATH_IS_SLASH(psz[0]))
147 return 0;
148 papszPaths[i] = RTPATH_STYLE_FN(rtPathHlpSkipSlashes)(psz);
149 }
150 pszPath0EndLastComp = pszPath0 = RTPATH_STYLE_FN(rtPathHlpSkipSlashes)(pszPath0);
151 }
152 }
153 /* Skip past the drive letter if there is one, as that eliminates the need
154 to handle ':' in the main loop below. */
155 else if ( RT_C_IS_ALPHA(pszPath0[0])
156 && pszPath0[1] == ':')
157 {
158 /* Drive letter part first: */
159 char const chDrv = RT_C_TO_UPPER(pszPath0[0]);
160 pszPath0 += 2;
161 pszPath0EndLastComp = pszPath0;
162
163 for (size_t i = 1; i < cPaths; i++)
164 {
165 const char *psz = papszPaths[i];
166 if ( ( psz[0] != chDrv
167 && RT_C_TO_UPPER(psz[0]) != chDrv)
168 || psz[1] != ':')
169 return 0;
170 papszPaths[i] = psz + 2;
171 }
172
173 /* Subsequent slashes or lack thereof. */
174 if (RTPATH_IS_SLASH(*pszPath0))
175 {
176 for (size_t i = 1; i < cPaths; i++)
177 {
178 const char *psz = papszPaths[i];
179 if (!RTPATH_IS_SLASH(*psz))
180 return pszPath0EndLastComp - pszPath0Start;
181 papszPaths[i] = RTPATH_STYLE_FN(rtPathHlpSkipSlashes)(psz);
182 }
183 pszPath0EndLastComp = pszPath0 = RTPATH_STYLE_FN(rtPathHlpSkipSlashes)(pszPath0);
184 }
185 else
186 for (size_t i = 1; i < cPaths; i++)
187 if (RTPATH_IS_SLASH(*papszPaths[i]))
188 return pszPath0EndLastComp - pszPath0Start;
189 }
190#endif
191
192 /*
193 * Main compare loop.
194 */
195 for (;;)
196 {
197 RTUNICP uc0;
198 int rc = RTStrGetCpEx(&pszPath0, &uc0);
199 AssertRCReturn(rc, 0);
200 if (!RTPATH_IS_SLASH(uc0))
201 {
202 if (uc0 != 0)
203 {
204 for (size_t i = 1; i < cPaths; i++)
205 {
206 RTUNICP uc;
207 rc = RTStrGetCpEx(&papszPaths[i], &uc);
208 AssertRCReturn(rc, 0);
209 if (uc == uc0)
210 { /* likely */}
211#if RTPATH_STYLE == RTPATH_STR_F_STYLE_DOS
212 else if ( RTUniCpToUpper(uc) == RTUniCpToUpper(uc0)
213 || RTUniCpToLower(uc) == RTUniCpToLower(uc0))
214 { /* less likely */}
215#endif
216 else
217 return pszPath0EndLastComp - pszPath0Start;
218 }
219 }
220 else
221 {
222 /* pszPath0 is at an end. Check the state of the others as we must
223 return the whole pszPath0 length if their are also at the end of
224 at a slash. */
225 for (size_t i = 1; i < cPaths; i++)
226 {
227 char ch = *papszPaths[i];
228 if ( ch != '\0'
229 && !RTPATH_IS_SLASH(ch))
230 return pszPath0EndLastComp - pszPath0Start;
231 }
232 return pszPath0 - 1 - pszPath0Start;
233 }
234 }
235 else
236 {
237 /* pszPath0 is at a slash. Check whether all the other are too or are at
238 the end of the string. If any other string ends here, we can return
239 the length up to but not including the slash. */
240 bool fDone = false;
241 for (size_t i = 1; i < cPaths; i++)
242 {
243 char ch = *papszPaths[i];
244 if (ch == '\0')
245 fDone = true;
246 else if (RTPATH_IS_SLASH(ch))
247 papszPaths[i] = RTPATH_STYLE_FN(rtPathHlpSkipSlashes)(papszPaths[i]);
248 else
249 return pszPath0EndLastComp - pszPath0Start;
250 }
251 if (fDone)
252 return pszPath0 - pszPath0Start;
253 pszPath0EndLastComp = pszPath0 = RTPATH_STYLE_FN(rtPathHlpSkipSlashes)(pszPath0 - 1);
254 }
255 }
256}
257
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